spikard 0.3.5 → 0.5.0
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/LICENSE +1 -1
- data/README.md +674 -659
- data/ext/spikard_rb/Cargo.toml +17 -17
- data/ext/spikard_rb/extconf.rb +10 -10
- data/ext/spikard_rb/src/lib.rs +6 -6
- data/lib/spikard/app.rb +405 -386
- data/lib/spikard/background.rb +27 -27
- data/lib/spikard/config.rb +396 -396
- data/lib/spikard/converters.rb +13 -13
- data/lib/spikard/handler_wrapper.rb +113 -113
- data/lib/spikard/provide.rb +214 -214
- data/lib/spikard/response.rb +173 -173
- data/lib/spikard/schema.rb +243 -243
- data/lib/spikard/sse.rb +111 -111
- data/lib/spikard/streaming_response.rb +44 -44
- data/lib/spikard/testing.rb +256 -221
- data/lib/spikard/upload_file.rb +131 -131
- data/lib/spikard/version.rb +5 -5
- data/lib/spikard/websocket.rb +59 -59
- data/lib/spikard.rb +43 -43
- data/sig/spikard.rbs +366 -360
- data/vendor/crates/spikard-bindings-shared/Cargo.toml +63 -0
- data/vendor/crates/spikard-bindings-shared/examples/config_extraction.rs +132 -0
- data/vendor/crates/spikard-bindings-shared/src/config_extractor.rs +752 -0
- data/vendor/crates/spikard-bindings-shared/src/conversion_traits.rs +194 -0
- data/vendor/crates/spikard-bindings-shared/src/di_traits.rs +246 -0
- data/vendor/crates/spikard-bindings-shared/src/error_response.rs +401 -0
- data/vendor/crates/spikard-bindings-shared/src/handler_base.rs +238 -0
- data/vendor/crates/spikard-bindings-shared/src/lib.rs +24 -0
- data/vendor/crates/spikard-bindings-shared/src/lifecycle_base.rs +292 -0
- data/vendor/crates/spikard-bindings-shared/src/lifecycle_executor.rs +616 -0
- data/vendor/crates/spikard-bindings-shared/src/response_builder.rs +305 -0
- data/vendor/crates/spikard-bindings-shared/src/test_client_base.rs +248 -0
- data/vendor/crates/spikard-bindings-shared/src/validation_helpers.rs +351 -0
- data/vendor/crates/spikard-bindings-shared/tests/comprehensive_coverage.rs +454 -0
- data/vendor/crates/spikard-bindings-shared/tests/error_response_edge_cases.rs +383 -0
- data/vendor/crates/spikard-bindings-shared/tests/handler_base_integration.rs +280 -0
- data/vendor/crates/spikard-core/Cargo.toml +40 -40
- data/vendor/crates/spikard-core/src/bindings/mod.rs +3 -3
- data/vendor/crates/spikard-core/src/bindings/response.rs +133 -133
- data/vendor/crates/spikard-core/src/debug.rs +127 -63
- data/vendor/crates/spikard-core/src/di/container.rs +702 -726
- data/vendor/crates/spikard-core/src/di/dependency.rs +273 -273
- data/vendor/crates/spikard-core/src/di/error.rs +118 -118
- data/vendor/crates/spikard-core/src/di/factory.rs +534 -538
- data/vendor/crates/spikard-core/src/di/graph.rs +506 -545
- data/vendor/crates/spikard-core/src/di/mod.rs +192 -192
- data/vendor/crates/spikard-core/src/di/resolved.rs +405 -411
- data/vendor/crates/spikard-core/src/di/value.rs +281 -283
- data/vendor/crates/spikard-core/src/errors.rs +69 -39
- data/vendor/crates/spikard-core/src/http.rs +415 -153
- data/vendor/crates/spikard-core/src/lib.rs +29 -29
- data/vendor/crates/spikard-core/src/lifecycle.rs +1186 -422
- data/vendor/crates/spikard-core/src/metadata.rs +389 -0
- data/vendor/crates/spikard-core/src/parameters.rs +2525 -722
- data/vendor/crates/spikard-core/src/problem.rs +344 -310
- data/vendor/crates/spikard-core/src/request_data.rs +1154 -189
- data/vendor/crates/spikard-core/src/router.rs +510 -249
- data/vendor/crates/spikard-core/src/schema_registry.rs +183 -183
- data/vendor/crates/spikard-core/src/type_hints.rs +304 -304
- data/vendor/crates/spikard-core/src/validation/error_mapper.rs +688 -0
- data/vendor/crates/spikard-core/src/{validation.rs → validation/mod.rs} +457 -699
- data/vendor/crates/spikard-http/Cargo.toml +64 -68
- data/vendor/crates/spikard-http/examples/sse-notifications.rs +148 -0
- data/vendor/crates/spikard-http/examples/websocket-chat.rs +92 -0
- data/vendor/crates/spikard-http/src/auth.rs +296 -247
- data/vendor/crates/spikard-http/src/background.rs +1860 -249
- data/vendor/crates/spikard-http/src/bindings/mod.rs +3 -3
- data/vendor/crates/spikard-http/src/bindings/response.rs +1 -1
- data/vendor/crates/spikard-http/src/body_metadata.rs +8 -8
- data/vendor/crates/spikard-http/src/cors.rs +1005 -490
- data/vendor/crates/spikard-http/src/debug.rs +128 -63
- data/vendor/crates/spikard-http/src/di_handler.rs +1668 -423
- data/vendor/crates/spikard-http/src/handler_response.rs +901 -190
- data/vendor/crates/spikard-http/src/handler_trait.rs +830 -228
- data/vendor/crates/spikard-http/src/handler_trait_tests.rs +290 -284
- data/vendor/crates/spikard-http/src/lib.rs +534 -529
- data/vendor/crates/spikard-http/src/lifecycle/adapter.rs +230 -149
- data/vendor/crates/spikard-http/src/lifecycle.rs +1193 -428
- data/vendor/crates/spikard-http/src/middleware/mod.rs +540 -285
- data/vendor/crates/spikard-http/src/middleware/multipart.rs +912 -86
- data/vendor/crates/spikard-http/src/middleware/urlencoded.rs +513 -147
- data/vendor/crates/spikard-http/src/middleware/validation.rs +735 -287
- data/vendor/crates/spikard-http/src/openapi/mod.rs +309 -309
- data/vendor/crates/spikard-http/src/openapi/parameter_extraction.rs +535 -190
- data/vendor/crates/spikard-http/src/openapi/schema_conversion.rs +1363 -308
- data/vendor/crates/spikard-http/src/openapi/spec_generation.rs +665 -195
- data/vendor/crates/spikard-http/src/query_parser.rs +793 -369
- data/vendor/crates/spikard-http/src/response.rs +720 -399
- data/vendor/crates/spikard-http/src/server/handler.rs +1650 -87
- data/vendor/crates/spikard-http/src/server/lifecycle_execution.rs +234 -98
- data/vendor/crates/spikard-http/src/server/mod.rs +1502 -805
- data/vendor/crates/spikard-http/src/server/request_extraction.rs +770 -119
- data/vendor/crates/spikard-http/src/server/routing_factory.rs +599 -0
- data/vendor/crates/spikard-http/src/sse.rs +1409 -447
- data/vendor/crates/spikard-http/src/testing/form.rs +52 -14
- data/vendor/crates/spikard-http/src/testing/multipart.rs +60 -60
- data/vendor/crates/spikard-http/src/testing/test_client.rs +283 -285
- data/vendor/crates/spikard-http/src/testing.rs +377 -377
- data/vendor/crates/spikard-http/src/websocket.rs +1375 -324
- data/vendor/crates/spikard-http/tests/background_behavior.rs +832 -0
- data/vendor/crates/spikard-http/tests/common/handlers.rs +309 -0
- data/vendor/crates/spikard-http/tests/common/mod.rs +26 -0
- data/vendor/crates/spikard-http/tests/di_integration.rs +192 -0
- data/vendor/crates/spikard-http/tests/doc_snippets.rs +5 -0
- data/vendor/crates/spikard-http/tests/lifecycle_execution.rs +1093 -0
- data/vendor/crates/spikard-http/tests/multipart_behavior.rs +656 -0
- data/vendor/crates/spikard-http/tests/server_config_builder.rs +314 -0
- data/vendor/crates/spikard-http/tests/sse_behavior.rs +620 -0
- data/vendor/crates/spikard-http/tests/websocket_behavior.rs +663 -0
- data/vendor/crates/spikard-rb/Cargo.toml +48 -42
- data/vendor/crates/spikard-rb/build.rs +199 -8
- data/vendor/crates/spikard-rb/src/background.rs +63 -63
- data/vendor/crates/spikard-rb/src/config/mod.rs +5 -0
- data/vendor/crates/spikard-rb/src/{config.rs → config/server_config.rs} +285 -294
- data/vendor/crates/spikard-rb/src/conversion.rs +554 -453
- data/vendor/crates/spikard-rb/src/di/builder.rs +100 -0
- data/vendor/crates/spikard-rb/src/{di.rs → di/mod.rs} +375 -409
- data/vendor/crates/spikard-rb/src/handler.rs +618 -625
- data/vendor/crates/spikard-rb/src/integration/mod.rs +3 -0
- data/vendor/crates/spikard-rb/src/lib.rs +1810 -2771
- data/vendor/crates/spikard-rb/src/lifecycle.rs +275 -274
- data/vendor/crates/spikard-rb/src/metadata/mod.rs +5 -0
- data/vendor/crates/spikard-rb/src/metadata/route_extraction.rs +447 -0
- data/vendor/crates/spikard-rb/src/runtime/mod.rs +5 -0
- data/vendor/crates/spikard-rb/src/runtime/server_runner.rs +324 -0
- data/vendor/crates/spikard-rb/src/server.rs +308 -283
- data/vendor/crates/spikard-rb/src/sse.rs +231 -231
- data/vendor/crates/spikard-rb/src/{test_client.rs → testing/client.rs} +551 -404
- data/vendor/crates/spikard-rb/src/testing/mod.rs +7 -0
- data/vendor/crates/spikard-rb/src/{test_sse.rs → testing/sse.rs} +143 -143
- data/vendor/crates/spikard-rb/src/testing/websocket.rs +635 -0
- data/vendor/crates/spikard-rb/src/websocket.rs +374 -233
- metadata +46 -13
- data/vendor/crates/spikard-http/src/parameters.rs +0 -1
- data/vendor/crates/spikard-http/src/problem.rs +0 -1
- data/vendor/crates/spikard-http/src/router.rs +0 -1
- data/vendor/crates/spikard-http/src/schema_registry.rs +0 -1
- data/vendor/crates/spikard-http/src/type_hints.rs +0 -1
- data/vendor/crates/spikard-http/src/validation.rs +0 -1
- data/vendor/crates/spikard-rb/src/test_websocket.rs +0 -221
|
@@ -1,39 +1,69 @@
|
|
|
1
|
-
//! Shared structured error types and panic shielding utilities.
|
|
2
|
-
//!
|
|
3
|
-
//! Bindings should convert all fatal paths into this shape to keep cross-language
|
|
4
|
-
//! error payloads consistent and avoid panics crossing FFI boundaries.
|
|
5
|
-
|
|
6
|
-
use serde::Serialize;
|
|
7
|
-
use serde_json::Value;
|
|
8
|
-
use std::panic::{UnwindSafe, catch_unwind};
|
|
9
|
-
|
|
10
|
-
/// Canonical error payload: { error, code, details }.
|
|
11
|
-
#[derive(Debug, Clone, Serialize)]
|
|
12
|
-
pub struct StructuredError {
|
|
13
|
-
pub error: String,
|
|
14
|
-
pub code: String,
|
|
15
|
-
#[serde(default)]
|
|
16
|
-
pub details: Value,
|
|
17
|
-
}
|
|
18
|
-
|
|
19
|
-
impl StructuredError {
|
|
20
|
-
pub fn new(code: impl Into<String>, error: impl Into<String>, details: Value) -> Self {
|
|
21
|
-
Self {
|
|
22
|
-
code: code.into(),
|
|
23
|
-
error: error.into(),
|
|
24
|
-
details,
|
|
25
|
-
}
|
|
26
|
-
}
|
|
27
|
-
|
|
28
|
-
pub fn simple(code: impl Into<String>, error: impl Into<String>) -> Self {
|
|
29
|
-
Self::new(code, error, Value::Object(serde_json::Map::new()))
|
|
30
|
-
}
|
|
31
|
-
}
|
|
32
|
-
|
|
33
|
-
/// Catch panics and convert to a structured error so they don't cross FFI boundaries.
|
|
34
|
-
pub fn shield_panic<T, F>(f: F) -> Result<T, StructuredError>
|
|
35
|
-
where
|
|
36
|
-
F: FnOnce() -> T + UnwindSafe,
|
|
37
|
-
{
|
|
38
|
-
catch_unwind(f).map_err(|_| StructuredError::simple("panic", "Unexpected panic in Rust code"))
|
|
39
|
-
}
|
|
1
|
+
//! Shared structured error types and panic shielding utilities.
|
|
2
|
+
//!
|
|
3
|
+
//! Bindings should convert all fatal paths into this shape to keep cross-language
|
|
4
|
+
//! error payloads consistent and avoid panics crossing FFI boundaries.
|
|
5
|
+
|
|
6
|
+
use serde::Serialize;
|
|
7
|
+
use serde_json::Value;
|
|
8
|
+
use std::panic::{UnwindSafe, catch_unwind};
|
|
9
|
+
|
|
10
|
+
/// Canonical error payload: { error, code, details }.
|
|
11
|
+
#[derive(Debug, Clone, Serialize)]
|
|
12
|
+
pub struct StructuredError {
|
|
13
|
+
pub error: String,
|
|
14
|
+
pub code: String,
|
|
15
|
+
#[serde(default)]
|
|
16
|
+
pub details: Value,
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
impl StructuredError {
|
|
20
|
+
pub fn new(code: impl Into<String>, error: impl Into<String>, details: Value) -> Self {
|
|
21
|
+
Self {
|
|
22
|
+
code: code.into(),
|
|
23
|
+
error: error.into(),
|
|
24
|
+
details,
|
|
25
|
+
}
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
pub fn simple(code: impl Into<String>, error: impl Into<String>) -> Self {
|
|
29
|
+
Self::new(code, error, Value::Object(serde_json::Map::new()))
|
|
30
|
+
}
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
/// Catch panics and convert to a structured error so they don't cross FFI boundaries.
|
|
34
|
+
pub fn shield_panic<T, F>(f: F) -> Result<T, StructuredError>
|
|
35
|
+
where
|
|
36
|
+
F: FnOnce() -> T + UnwindSafe,
|
|
37
|
+
{
|
|
38
|
+
catch_unwind(f).map_err(|_| StructuredError::simple("panic", "Unexpected panic in Rust code"))
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
#[cfg(test)]
|
|
42
|
+
mod tests {
|
|
43
|
+
use super::*;
|
|
44
|
+
use serde_json::json;
|
|
45
|
+
|
|
46
|
+
#[test]
|
|
47
|
+
fn structured_error_constructors_populate_fields() {
|
|
48
|
+
let details = json!({"field": "name"});
|
|
49
|
+
let err = StructuredError::new("invalid", "bad input", details.clone());
|
|
50
|
+
assert_eq!(err.code, "invalid");
|
|
51
|
+
assert_eq!(err.error, "bad input");
|
|
52
|
+
assert_eq!(err.details, details);
|
|
53
|
+
|
|
54
|
+
let simple = StructuredError::simple("missing", "not found");
|
|
55
|
+
assert_eq!(simple.code, "missing");
|
|
56
|
+
assert_eq!(simple.error, "not found");
|
|
57
|
+
assert!(simple.details.is_object());
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
#[test]
|
|
61
|
+
fn shield_panic_returns_ok_or_structured_error() {
|
|
62
|
+
let ok = shield_panic(|| 42);
|
|
63
|
+
assert_eq!(ok.unwrap(), 42);
|
|
64
|
+
|
|
65
|
+
let err = shield_panic(|| panic!("boom")).unwrap_err();
|
|
66
|
+
assert_eq!(err.code, "panic");
|
|
67
|
+
assert!(err.error.contains("Unexpected panic"));
|
|
68
|
+
}
|
|
69
|
+
}
|
|
@@ -1,153 +1,415 @@
|
|
|
1
|
-
use serde::{Deserialize, Serialize};
|
|
2
|
-
use serde_json::Value;
|
|
3
|
-
|
|
4
|
-
/// HTTP method
|
|
5
|
-
#[derive(Debug, Clone, PartialEq, Eq, Hash, Serialize, Deserialize)]
|
|
6
|
-
pub enum Method {
|
|
7
|
-
Get,
|
|
8
|
-
Post,
|
|
9
|
-
Put,
|
|
10
|
-
Patch,
|
|
11
|
-
Delete,
|
|
12
|
-
Head,
|
|
13
|
-
Options,
|
|
14
|
-
Trace,
|
|
15
|
-
}
|
|
16
|
-
|
|
17
|
-
impl Method {
|
|
18
|
-
pub fn as_str(&self) -> &'static str {
|
|
19
|
-
match self {
|
|
20
|
-
Method::Get => "GET",
|
|
21
|
-
Method::Post => "POST",
|
|
22
|
-
Method::Put => "PUT",
|
|
23
|
-
Method::Patch => "PATCH",
|
|
24
|
-
Method::Delete => "DELETE",
|
|
25
|
-
Method::Head => "HEAD",
|
|
26
|
-
Method::Options => "OPTIONS",
|
|
27
|
-
Method::Trace => "TRACE",
|
|
28
|
-
}
|
|
29
|
-
}
|
|
30
|
-
}
|
|
31
|
-
|
|
32
|
-
impl std::fmt::Display for Method {
|
|
33
|
-
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
|
34
|
-
write!(f, "{}", self.as_str())
|
|
35
|
-
}
|
|
36
|
-
}
|
|
37
|
-
|
|
38
|
-
impl std::str::FromStr for Method {
|
|
39
|
-
type Err = String;
|
|
40
|
-
|
|
41
|
-
fn from_str(s: &str) -> Result<Self, Self::Err> {
|
|
42
|
-
match s.to_uppercase().as_str() {
|
|
43
|
-
"GET" => Ok(Method::Get),
|
|
44
|
-
"POST" => Ok(Method::Post),
|
|
45
|
-
"PUT" => Ok(Method::Put),
|
|
46
|
-
"PATCH" => Ok(Method::Patch),
|
|
47
|
-
"DELETE" => Ok(Method::Delete),
|
|
48
|
-
"HEAD" => Ok(Method::Head),
|
|
49
|
-
"OPTIONS" => Ok(Method::Options),
|
|
50
|
-
"TRACE" => Ok(Method::Trace),
|
|
51
|
-
_ => Err(format!("Unknown HTTP method: {}", s)),
|
|
52
|
-
}
|
|
53
|
-
}
|
|
54
|
-
}
|
|
55
|
-
|
|
56
|
-
/// CORS configuration for a route
|
|
57
|
-
#[derive(Debug, Clone, Serialize, Deserialize)]
|
|
58
|
-
pub struct CorsConfig {
|
|
59
|
-
pub allowed_origins: Vec<String>,
|
|
60
|
-
pub allowed_methods: Vec<String>,
|
|
61
|
-
#[serde(default)]
|
|
62
|
-
pub allowed_headers: Vec<String>,
|
|
63
|
-
#[serde(skip_serializing_if = "Option::is_none")]
|
|
64
|
-
pub expose_headers: Option<Vec<String>>,
|
|
65
|
-
#[serde(skip_serializing_if = "Option::is_none")]
|
|
66
|
-
pub max_age: Option<u32>,
|
|
67
|
-
#[serde(skip_serializing_if = "Option::is_none")]
|
|
68
|
-
pub allow_credentials: Option<bool>,
|
|
69
|
-
}
|
|
70
|
-
|
|
71
|
-
/// Route metadata extracted from bindings
|
|
72
|
-
#[derive(Debug, Clone, Serialize, Deserialize)]
|
|
73
|
-
pub struct RouteMetadata {
|
|
74
|
-
pub method: String,
|
|
75
|
-
pub path: String,
|
|
76
|
-
pub handler_name: String,
|
|
77
|
-
pub request_schema: Option<Value>,
|
|
78
|
-
pub response_schema: Option<Value>,
|
|
79
|
-
pub parameter_schema: Option<Value>,
|
|
80
|
-
#[serde(skip_serializing_if = "Option::is_none")]
|
|
81
|
-
pub file_params: Option<Value>,
|
|
82
|
-
pub is_async: bool,
|
|
83
|
-
pub cors: Option<CorsConfig>,
|
|
84
|
-
/// Name of the body parameter (defaults to "body" if not specified)
|
|
85
|
-
#[serde(skip_serializing_if = "Option::is_none")]
|
|
86
|
-
pub body_param_name: Option<String>,
|
|
87
|
-
/// List of dependency keys this handler requires (for DI)
|
|
88
|
-
#[cfg(feature = "di")]
|
|
89
|
-
#[serde(skip_serializing_if = "Option::is_none")]
|
|
90
|
-
pub handler_dependencies: Option<Vec<String>>,
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
/// Enable
|
|
100
|
-
#[serde(default = "default_true")]
|
|
101
|
-
pub
|
|
102
|
-
///
|
|
103
|
-
#[serde(default = "
|
|
104
|
-
pub
|
|
105
|
-
///
|
|
106
|
-
#[serde(default = "
|
|
107
|
-
pub
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
129
|
-
|
|
130
|
-
|
|
131
|
-
|
|
132
|
-
|
|
133
|
-
|
|
134
|
-
|
|
135
|
-
|
|
136
|
-
|
|
137
|
-
|
|
138
|
-
|
|
139
|
-
|
|
140
|
-
|
|
141
|
-
|
|
142
|
-
pub
|
|
143
|
-
|
|
144
|
-
|
|
145
|
-
|
|
146
|
-
|
|
147
|
-
|
|
148
|
-
|
|
149
|
-
|
|
150
|
-
|
|
151
|
-
|
|
152
|
-
|
|
153
|
-
|
|
1
|
+
use serde::{Deserialize, Serialize};
|
|
2
|
+
use serde_json::Value;
|
|
3
|
+
|
|
4
|
+
/// HTTP method
|
|
5
|
+
#[derive(Debug, Clone, PartialEq, Eq, Hash, Serialize, Deserialize)]
|
|
6
|
+
pub enum Method {
|
|
7
|
+
Get,
|
|
8
|
+
Post,
|
|
9
|
+
Put,
|
|
10
|
+
Patch,
|
|
11
|
+
Delete,
|
|
12
|
+
Head,
|
|
13
|
+
Options,
|
|
14
|
+
Trace,
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
impl Method {
|
|
18
|
+
pub fn as_str(&self) -> &'static str {
|
|
19
|
+
match self {
|
|
20
|
+
Method::Get => "GET",
|
|
21
|
+
Method::Post => "POST",
|
|
22
|
+
Method::Put => "PUT",
|
|
23
|
+
Method::Patch => "PATCH",
|
|
24
|
+
Method::Delete => "DELETE",
|
|
25
|
+
Method::Head => "HEAD",
|
|
26
|
+
Method::Options => "OPTIONS",
|
|
27
|
+
Method::Trace => "TRACE",
|
|
28
|
+
}
|
|
29
|
+
}
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
impl std::fmt::Display for Method {
|
|
33
|
+
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
|
34
|
+
write!(f, "{}", self.as_str())
|
|
35
|
+
}
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
impl std::str::FromStr for Method {
|
|
39
|
+
type Err = String;
|
|
40
|
+
|
|
41
|
+
fn from_str(s: &str) -> Result<Self, Self::Err> {
|
|
42
|
+
match s.to_uppercase().as_str() {
|
|
43
|
+
"GET" => Ok(Method::Get),
|
|
44
|
+
"POST" => Ok(Method::Post),
|
|
45
|
+
"PUT" => Ok(Method::Put),
|
|
46
|
+
"PATCH" => Ok(Method::Patch),
|
|
47
|
+
"DELETE" => Ok(Method::Delete),
|
|
48
|
+
"HEAD" => Ok(Method::Head),
|
|
49
|
+
"OPTIONS" => Ok(Method::Options),
|
|
50
|
+
"TRACE" => Ok(Method::Trace),
|
|
51
|
+
_ => Err(format!("Unknown HTTP method: {}", s)),
|
|
52
|
+
}
|
|
53
|
+
}
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
/// CORS configuration for a route
|
|
57
|
+
#[derive(Debug, Clone, Serialize, Deserialize)]
|
|
58
|
+
pub struct CorsConfig {
|
|
59
|
+
pub allowed_origins: Vec<String>,
|
|
60
|
+
pub allowed_methods: Vec<String>,
|
|
61
|
+
#[serde(default)]
|
|
62
|
+
pub allowed_headers: Vec<String>,
|
|
63
|
+
#[serde(skip_serializing_if = "Option::is_none")]
|
|
64
|
+
pub expose_headers: Option<Vec<String>>,
|
|
65
|
+
#[serde(skip_serializing_if = "Option::is_none")]
|
|
66
|
+
pub max_age: Option<u32>,
|
|
67
|
+
#[serde(skip_serializing_if = "Option::is_none")]
|
|
68
|
+
pub allow_credentials: Option<bool>,
|
|
69
|
+
}
|
|
70
|
+
|
|
71
|
+
/// Route metadata extracted from bindings
|
|
72
|
+
#[derive(Debug, Clone, Serialize, Deserialize)]
|
|
73
|
+
pub struct RouteMetadata {
|
|
74
|
+
pub method: String,
|
|
75
|
+
pub path: String,
|
|
76
|
+
pub handler_name: String,
|
|
77
|
+
pub request_schema: Option<Value>,
|
|
78
|
+
pub response_schema: Option<Value>,
|
|
79
|
+
pub parameter_schema: Option<Value>,
|
|
80
|
+
#[serde(skip_serializing_if = "Option::is_none")]
|
|
81
|
+
pub file_params: Option<Value>,
|
|
82
|
+
pub is_async: bool,
|
|
83
|
+
pub cors: Option<CorsConfig>,
|
|
84
|
+
/// Name of the body parameter (defaults to "body" if not specified)
|
|
85
|
+
#[serde(skip_serializing_if = "Option::is_none")]
|
|
86
|
+
pub body_param_name: Option<String>,
|
|
87
|
+
/// List of dependency keys this handler requires (for DI)
|
|
88
|
+
#[cfg(feature = "di")]
|
|
89
|
+
#[serde(skip_serializing_if = "Option::is_none")]
|
|
90
|
+
pub handler_dependencies: Option<Vec<String>>,
|
|
91
|
+
/// JSON-RPC method metadata (if this route is exposed as a JSON-RPC method)
|
|
92
|
+
#[serde(skip_serializing_if = "Option::is_none")]
|
|
93
|
+
pub jsonrpc_method: Option<Value>,
|
|
94
|
+
}
|
|
95
|
+
|
|
96
|
+
/// Compression configuration shared across runtimes
|
|
97
|
+
#[derive(Debug, Clone, Serialize, Deserialize)]
|
|
98
|
+
pub struct CompressionConfig {
|
|
99
|
+
/// Enable gzip compression
|
|
100
|
+
#[serde(default = "default_true")]
|
|
101
|
+
pub gzip: bool,
|
|
102
|
+
/// Enable brotli compression
|
|
103
|
+
#[serde(default = "default_true")]
|
|
104
|
+
pub brotli: bool,
|
|
105
|
+
/// Minimum response size to compress (bytes)
|
|
106
|
+
#[serde(default = "default_compression_min_size")]
|
|
107
|
+
pub min_size: usize,
|
|
108
|
+
/// Compression quality (0-11 for brotli, 0-9 for gzip)
|
|
109
|
+
#[serde(default = "default_compression_quality")]
|
|
110
|
+
pub quality: u32,
|
|
111
|
+
}
|
|
112
|
+
|
|
113
|
+
const fn default_true() -> bool {
|
|
114
|
+
true
|
|
115
|
+
}
|
|
116
|
+
|
|
117
|
+
const fn default_compression_min_size() -> usize {
|
|
118
|
+
1024
|
|
119
|
+
}
|
|
120
|
+
|
|
121
|
+
const fn default_compression_quality() -> u32 {
|
|
122
|
+
6
|
|
123
|
+
}
|
|
124
|
+
|
|
125
|
+
impl Default for CompressionConfig {
|
|
126
|
+
fn default() -> Self {
|
|
127
|
+
Self {
|
|
128
|
+
gzip: true,
|
|
129
|
+
brotli: true,
|
|
130
|
+
min_size: default_compression_min_size(),
|
|
131
|
+
quality: default_compression_quality(),
|
|
132
|
+
}
|
|
133
|
+
}
|
|
134
|
+
}
|
|
135
|
+
|
|
136
|
+
/// Rate limiting configuration shared across runtimes
|
|
137
|
+
#[derive(Debug, Clone, Serialize, Deserialize)]
|
|
138
|
+
pub struct RateLimitConfig {
|
|
139
|
+
/// Requests per second
|
|
140
|
+
pub per_second: u64,
|
|
141
|
+
/// Burst allowance
|
|
142
|
+
pub burst: u32,
|
|
143
|
+
/// Use IP-based rate limiting
|
|
144
|
+
#[serde(default = "default_true")]
|
|
145
|
+
pub ip_based: bool,
|
|
146
|
+
}
|
|
147
|
+
|
|
148
|
+
impl Default for RateLimitConfig {
|
|
149
|
+
fn default() -> Self {
|
|
150
|
+
Self {
|
|
151
|
+
per_second: 100,
|
|
152
|
+
burst: 200,
|
|
153
|
+
ip_based: true,
|
|
154
|
+
}
|
|
155
|
+
}
|
|
156
|
+
}
|
|
157
|
+
|
|
158
|
+
#[cfg(test)]
|
|
159
|
+
mod tests {
|
|
160
|
+
use super::*;
|
|
161
|
+
use std::str::FromStr;
|
|
162
|
+
|
|
163
|
+
#[test]
|
|
164
|
+
fn test_method_as_str_get() {
|
|
165
|
+
assert_eq!(Method::Get.as_str(), "GET");
|
|
166
|
+
}
|
|
167
|
+
|
|
168
|
+
#[test]
|
|
169
|
+
fn test_method_as_str_post() {
|
|
170
|
+
assert_eq!(Method::Post.as_str(), "POST");
|
|
171
|
+
}
|
|
172
|
+
|
|
173
|
+
#[test]
|
|
174
|
+
fn test_method_as_str_put() {
|
|
175
|
+
assert_eq!(Method::Put.as_str(), "PUT");
|
|
176
|
+
}
|
|
177
|
+
|
|
178
|
+
#[test]
|
|
179
|
+
fn test_method_as_str_patch() {
|
|
180
|
+
assert_eq!(Method::Patch.as_str(), "PATCH");
|
|
181
|
+
}
|
|
182
|
+
|
|
183
|
+
#[test]
|
|
184
|
+
fn test_method_as_str_delete() {
|
|
185
|
+
assert_eq!(Method::Delete.as_str(), "DELETE");
|
|
186
|
+
}
|
|
187
|
+
|
|
188
|
+
#[test]
|
|
189
|
+
fn test_method_as_str_head() {
|
|
190
|
+
assert_eq!(Method::Head.as_str(), "HEAD");
|
|
191
|
+
}
|
|
192
|
+
|
|
193
|
+
#[test]
|
|
194
|
+
fn test_method_as_str_options() {
|
|
195
|
+
assert_eq!(Method::Options.as_str(), "OPTIONS");
|
|
196
|
+
}
|
|
197
|
+
|
|
198
|
+
#[test]
|
|
199
|
+
fn test_method_as_str_trace() {
|
|
200
|
+
assert_eq!(Method::Trace.as_str(), "TRACE");
|
|
201
|
+
}
|
|
202
|
+
|
|
203
|
+
#[test]
|
|
204
|
+
fn test_method_display_get() {
|
|
205
|
+
assert_eq!(Method::Get.to_string(), "GET");
|
|
206
|
+
}
|
|
207
|
+
|
|
208
|
+
#[test]
|
|
209
|
+
fn test_method_display_post() {
|
|
210
|
+
assert_eq!(Method::Post.to_string(), "POST");
|
|
211
|
+
}
|
|
212
|
+
|
|
213
|
+
#[test]
|
|
214
|
+
fn test_method_display_put() {
|
|
215
|
+
assert_eq!(Method::Put.to_string(), "PUT");
|
|
216
|
+
}
|
|
217
|
+
|
|
218
|
+
#[test]
|
|
219
|
+
fn test_method_display_patch() {
|
|
220
|
+
assert_eq!(Method::Patch.to_string(), "PATCH");
|
|
221
|
+
}
|
|
222
|
+
|
|
223
|
+
#[test]
|
|
224
|
+
fn test_method_display_delete() {
|
|
225
|
+
assert_eq!(Method::Delete.to_string(), "DELETE");
|
|
226
|
+
}
|
|
227
|
+
|
|
228
|
+
#[test]
|
|
229
|
+
fn test_method_display_head() {
|
|
230
|
+
assert_eq!(Method::Head.to_string(), "HEAD");
|
|
231
|
+
}
|
|
232
|
+
|
|
233
|
+
#[test]
|
|
234
|
+
fn test_method_display_options() {
|
|
235
|
+
assert_eq!(Method::Options.to_string(), "OPTIONS");
|
|
236
|
+
}
|
|
237
|
+
|
|
238
|
+
#[test]
|
|
239
|
+
fn test_method_display_trace() {
|
|
240
|
+
assert_eq!(Method::Trace.to_string(), "TRACE");
|
|
241
|
+
}
|
|
242
|
+
|
|
243
|
+
#[test]
|
|
244
|
+
fn test_from_str_get() {
|
|
245
|
+
assert_eq!(Method::from_str("GET"), Ok(Method::Get));
|
|
246
|
+
}
|
|
247
|
+
|
|
248
|
+
#[test]
|
|
249
|
+
fn test_from_str_post() {
|
|
250
|
+
assert_eq!(Method::from_str("POST"), Ok(Method::Post));
|
|
251
|
+
}
|
|
252
|
+
|
|
253
|
+
#[test]
|
|
254
|
+
fn test_from_str_put() {
|
|
255
|
+
assert_eq!(Method::from_str("PUT"), Ok(Method::Put));
|
|
256
|
+
}
|
|
257
|
+
|
|
258
|
+
#[test]
|
|
259
|
+
fn test_from_str_patch() {
|
|
260
|
+
assert_eq!(Method::from_str("PATCH"), Ok(Method::Patch));
|
|
261
|
+
}
|
|
262
|
+
|
|
263
|
+
#[test]
|
|
264
|
+
fn test_from_str_delete() {
|
|
265
|
+
assert_eq!(Method::from_str("DELETE"), Ok(Method::Delete));
|
|
266
|
+
}
|
|
267
|
+
|
|
268
|
+
#[test]
|
|
269
|
+
fn test_from_str_head() {
|
|
270
|
+
assert_eq!(Method::from_str("HEAD"), Ok(Method::Head));
|
|
271
|
+
}
|
|
272
|
+
|
|
273
|
+
#[test]
|
|
274
|
+
fn test_from_str_options() {
|
|
275
|
+
assert_eq!(Method::from_str("OPTIONS"), Ok(Method::Options));
|
|
276
|
+
}
|
|
277
|
+
|
|
278
|
+
#[test]
|
|
279
|
+
fn test_from_str_trace() {
|
|
280
|
+
assert_eq!(Method::from_str("TRACE"), Ok(Method::Trace));
|
|
281
|
+
}
|
|
282
|
+
|
|
283
|
+
#[test]
|
|
284
|
+
fn test_from_str_lowercase() {
|
|
285
|
+
assert_eq!(Method::from_str("get"), Ok(Method::Get));
|
|
286
|
+
}
|
|
287
|
+
|
|
288
|
+
#[test]
|
|
289
|
+
fn test_from_str_mixed_case() {
|
|
290
|
+
assert_eq!(Method::from_str("PoSt"), Ok(Method::Post));
|
|
291
|
+
}
|
|
292
|
+
|
|
293
|
+
#[test]
|
|
294
|
+
fn test_from_str_invalid_method() {
|
|
295
|
+
let result = Method::from_str("INVALID");
|
|
296
|
+
assert!(result.is_err());
|
|
297
|
+
assert_eq!(result.unwrap_err(), "Unknown HTTP method: INVALID");
|
|
298
|
+
}
|
|
299
|
+
|
|
300
|
+
#[test]
|
|
301
|
+
fn test_from_str_empty_string() {
|
|
302
|
+
let result = Method::from_str("");
|
|
303
|
+
assert!(result.is_err());
|
|
304
|
+
assert_eq!(result.unwrap_err(), "Unknown HTTP method: ");
|
|
305
|
+
}
|
|
306
|
+
|
|
307
|
+
#[test]
|
|
308
|
+
fn test_compression_config_default() {
|
|
309
|
+
let config = CompressionConfig::default();
|
|
310
|
+
assert!(config.gzip);
|
|
311
|
+
assert!(config.brotli);
|
|
312
|
+
assert_eq!(config.min_size, 1024);
|
|
313
|
+
assert_eq!(config.quality, 6);
|
|
314
|
+
}
|
|
315
|
+
|
|
316
|
+
#[test]
|
|
317
|
+
fn test_default_true() {
|
|
318
|
+
assert!(default_true());
|
|
319
|
+
}
|
|
320
|
+
|
|
321
|
+
#[test]
|
|
322
|
+
fn test_default_compression_min_size() {
|
|
323
|
+
assert_eq!(default_compression_min_size(), 1024);
|
|
324
|
+
}
|
|
325
|
+
|
|
326
|
+
#[test]
|
|
327
|
+
fn test_default_compression_quality() {
|
|
328
|
+
assert_eq!(default_compression_quality(), 6);
|
|
329
|
+
}
|
|
330
|
+
|
|
331
|
+
#[test]
|
|
332
|
+
fn test_rate_limit_config_default() {
|
|
333
|
+
let config = RateLimitConfig::default();
|
|
334
|
+
assert_eq!(config.per_second, 100);
|
|
335
|
+
assert_eq!(config.burst, 200);
|
|
336
|
+
assert!(config.ip_based);
|
|
337
|
+
}
|
|
338
|
+
|
|
339
|
+
#[test]
|
|
340
|
+
fn test_method_equality() {
|
|
341
|
+
assert_eq!(Method::Get, Method::Get);
|
|
342
|
+
assert_ne!(Method::Get, Method::Post);
|
|
343
|
+
}
|
|
344
|
+
|
|
345
|
+
#[test]
|
|
346
|
+
fn test_method_clone() {
|
|
347
|
+
let method = Method::Post;
|
|
348
|
+
let cloned = method.clone();
|
|
349
|
+
assert_eq!(method, cloned);
|
|
350
|
+
}
|
|
351
|
+
|
|
352
|
+
#[test]
|
|
353
|
+
fn test_compression_config_custom_values() {
|
|
354
|
+
let config = CompressionConfig {
|
|
355
|
+
gzip: false,
|
|
356
|
+
brotli: false,
|
|
357
|
+
min_size: 2048,
|
|
358
|
+
quality: 11,
|
|
359
|
+
};
|
|
360
|
+
assert!(!config.gzip);
|
|
361
|
+
assert!(!config.brotli);
|
|
362
|
+
assert_eq!(config.min_size, 2048);
|
|
363
|
+
assert_eq!(config.quality, 11);
|
|
364
|
+
}
|
|
365
|
+
|
|
366
|
+
#[test]
|
|
367
|
+
fn test_rate_limit_config_custom_values() {
|
|
368
|
+
let config = RateLimitConfig {
|
|
369
|
+
per_second: 50,
|
|
370
|
+
burst: 100,
|
|
371
|
+
ip_based: false,
|
|
372
|
+
};
|
|
373
|
+
assert_eq!(config.per_second, 50);
|
|
374
|
+
assert_eq!(config.burst, 100);
|
|
375
|
+
assert!(!config.ip_based);
|
|
376
|
+
}
|
|
377
|
+
|
|
378
|
+
#[test]
|
|
379
|
+
fn test_cors_config_construction() {
|
|
380
|
+
let cors = CorsConfig {
|
|
381
|
+
allowed_origins: vec!["http://localhost:3000".to_string()],
|
|
382
|
+
allowed_methods: vec!["GET".to_string(), "POST".to_string()],
|
|
383
|
+
allowed_headers: vec![],
|
|
384
|
+
expose_headers: None,
|
|
385
|
+
max_age: None,
|
|
386
|
+
allow_credentials: None,
|
|
387
|
+
};
|
|
388
|
+
assert_eq!(cors.allowed_origins.len(), 1);
|
|
389
|
+
assert_eq!(cors.allowed_methods.len(), 2);
|
|
390
|
+
assert_eq!(cors.allowed_headers.len(), 0);
|
|
391
|
+
}
|
|
392
|
+
|
|
393
|
+
#[test]
|
|
394
|
+
fn test_route_metadata_construction() {
|
|
395
|
+
let metadata = RouteMetadata {
|
|
396
|
+
method: "GET".to_string(),
|
|
397
|
+
path: "/api/users".to_string(),
|
|
398
|
+
handler_name: "get_users".to_string(),
|
|
399
|
+
request_schema: None,
|
|
400
|
+
response_schema: None,
|
|
401
|
+
parameter_schema: None,
|
|
402
|
+
file_params: None,
|
|
403
|
+
is_async: true,
|
|
404
|
+
cors: None,
|
|
405
|
+
body_param_name: None,
|
|
406
|
+
#[cfg(feature = "di")]
|
|
407
|
+
handler_dependencies: None,
|
|
408
|
+
jsonrpc_method: None,
|
|
409
|
+
};
|
|
410
|
+
assert_eq!(metadata.method, "GET");
|
|
411
|
+
assert_eq!(metadata.path, "/api/users");
|
|
412
|
+
assert_eq!(metadata.handler_name, "get_users");
|
|
413
|
+
assert!(metadata.is_async);
|
|
414
|
+
}
|
|
415
|
+
}
|