spikard 0.12.0 → 0.15.2
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/Steepfile +6 -0
- data/ext/spikard_rb/extconf.rb +1 -2
- data/ext/spikard_rb/{Cargo.lock → native/Cargo.lock} +897 -451
- data/ext/spikard_rb/native/Cargo.toml +24 -0
- data/ext/spikard_rb/src/lib.rs +5366 -3
- data/lib/spikard/native.rb +86 -0
- data/lib/spikard/version.rb +6 -1
- data/lib/spikard.rb +8 -45
- data/lib/spikard_rb.so +0 -0
- data/sig/types.rbs +427 -0
- metadata +14 -242
- data/LICENSE +0 -1
- data/README.md +0 -267
- data/ext/spikard_rb/Cargo.toml +0 -17
- data/lib/spikard/app.rb +0 -428
- data/lib/spikard/background.rb +0 -58
- data/lib/spikard/config.rb +0 -506
- data/lib/spikard/converters.rb +0 -13
- data/lib/spikard/grpc.rb +0 -182
- data/lib/spikard/handler_wrapper.rb +0 -113
- data/lib/spikard/provide.rb +0 -214
- data/lib/spikard/response.rb +0 -173
- data/lib/spikard/schema.rb +0 -243
- data/lib/spikard/sse.rb +0 -111
- data/lib/spikard/streaming_response.rb +0 -44
- data/lib/spikard/testing.rb +0 -432
- data/lib/spikard/upload_file.rb +0 -131
- data/lib/spikard/websocket.rb +0 -59
- data/sig/spikard.rbs +0 -719
- data/vendor/crates/spikard-bindings-shared/Cargo.toml +0 -80
- data/vendor/crates/spikard-bindings-shared/examples/config_extraction.rs +0 -132
- data/vendor/crates/spikard-bindings-shared/src/config_extractor.rs +0 -905
- data/vendor/crates/spikard-bindings-shared/src/conversion_traits.rs +0 -210
- data/vendor/crates/spikard-bindings-shared/src/di_traits.rs +0 -252
- data/vendor/crates/spikard-bindings-shared/src/error_response.rs +0 -404
- data/vendor/crates/spikard-bindings-shared/src/grpc_metadata.rs +0 -199
- data/vendor/crates/spikard-bindings-shared/src/handler_base.rs +0 -252
- data/vendor/crates/spikard-bindings-shared/src/json_conversion.rs +0 -829
- data/vendor/crates/spikard-bindings-shared/src/lazy_cache.rs +0 -587
- data/vendor/crates/spikard-bindings-shared/src/lib.rs +0 -33
- data/vendor/crates/spikard-bindings-shared/src/lifecycle_base.rs +0 -298
- data/vendor/crates/spikard-bindings-shared/src/lifecycle_executor.rs +0 -594
- data/vendor/crates/spikard-bindings-shared/src/response_builder.rs +0 -743
- data/vendor/crates/spikard-bindings-shared/src/response_interpreter.rs +0 -944
- data/vendor/crates/spikard-bindings-shared/src/test_client_base.rs +0 -260
- data/vendor/crates/spikard-bindings-shared/src/validation_helpers.rs +0 -369
- data/vendor/crates/spikard-bindings-shared/tests/config_extractor_behavior.rs +0 -192
- data/vendor/crates/spikard-bindings-shared/tests/error_response_edge_cases.rs +0 -383
- data/vendor/crates/spikard-bindings-shared/tests/full_coverage.rs +0 -459
- data/vendor/crates/spikard-bindings-shared/tests/handler_base_integration.rs +0 -280
- data/vendor/crates/spikard-bindings-shared/tests/integration_tests.rs +0 -669
- data/vendor/crates/spikard-core/Cargo.toml +0 -60
- data/vendor/crates/spikard-core/src/bindings/mod.rs +0 -3
- data/vendor/crates/spikard-core/src/bindings/response.rs +0 -130
- data/vendor/crates/spikard-core/src/debug.rs +0 -127
- data/vendor/crates/spikard-core/src/di/container.rs +0 -702
- data/vendor/crates/spikard-core/src/di/dependency.rs +0 -273
- data/vendor/crates/spikard-core/src/di/error.rs +0 -118
- data/vendor/crates/spikard-core/src/di/factory.rs +0 -538
- data/vendor/crates/spikard-core/src/di/graph.rs +0 -507
- data/vendor/crates/spikard-core/src/di/mod.rs +0 -192
- data/vendor/crates/spikard-core/src/di/resolved.rs +0 -428
- data/vendor/crates/spikard-core/src/di/value.rs +0 -282
- data/vendor/crates/spikard-core/src/errors.rs +0 -72
- data/vendor/crates/spikard-core/src/http.rs +0 -492
- data/vendor/crates/spikard-core/src/lib.rs +0 -29
- data/vendor/crates/spikard-core/src/lifecycle.rs +0 -1273
- data/vendor/crates/spikard-core/src/metadata.rs +0 -378
- data/vendor/crates/spikard-core/src/parameters.rs +0 -2546
- data/vendor/crates/spikard-core/src/problem.rs +0 -358
- data/vendor/crates/spikard-core/src/request_data.rs +0 -1146
- data/vendor/crates/spikard-core/src/router.rs +0 -530
- data/vendor/crates/spikard-core/src/schema_registry.rs +0 -197
- data/vendor/crates/spikard-core/src/type_hints.rs +0 -311
- data/vendor/crates/spikard-core/src/validation/error_mapper.rs +0 -710
- data/vendor/crates/spikard-core/src/validation/mod.rs +0 -470
- data/vendor/crates/spikard-core/tests/bindings_response_tests.rs +0 -136
- data/vendor/crates/spikard-core/tests/di_dependency_defaults.rs +0 -37
- data/vendor/crates/spikard-core/tests/error_mapper.rs +0 -761
- data/vendor/crates/spikard-core/tests/parameters_edge_cases.rs +0 -106
- data/vendor/crates/spikard-core/tests/parameters_full.rs +0 -701
- data/vendor/crates/spikard-core/tests/parameters_schema_and_formats.rs +0 -301
- data/vendor/crates/spikard-core/tests/request_data_roundtrip.rs +0 -67
- data/vendor/crates/spikard-core/tests/validation_coverage.rs +0 -250
- data/vendor/crates/spikard-core/tests/validation_error_paths.rs +0 -45
- data/vendor/crates/spikard-http/Cargo.toml +0 -87
- data/vendor/crates/spikard-http/examples/sse-notifications.rs +0 -148
- data/vendor/crates/spikard-http/examples/websocket-chat.rs +0 -92
- data/vendor/crates/spikard-http/src/auth.rs +0 -301
- data/vendor/crates/spikard-http/src/background.rs +0 -1860
- data/vendor/crates/spikard-http/src/bindings/mod.rs +0 -3
- data/vendor/crates/spikard-http/src/bindings/response.rs +0 -1
- data/vendor/crates/spikard-http/src/body_metadata.rs +0 -8
- data/vendor/crates/spikard-http/src/cors.rs +0 -1026
- data/vendor/crates/spikard-http/src/debug.rs +0 -128
- data/vendor/crates/spikard-http/src/di_handler.rs +0 -1672
- data/vendor/crates/spikard-http/src/grpc/framing.rs +0 -469
- data/vendor/crates/spikard-http/src/grpc/handler.rs +0 -1122
- data/vendor/crates/spikard-http/src/grpc/mod.rs +0 -434
- data/vendor/crates/spikard-http/src/grpc/service.rs +0 -622
- data/vendor/crates/spikard-http/src/grpc/streaming.rs +0 -319
- data/vendor/crates/spikard-http/src/handler_response.rs +0 -901
- data/vendor/crates/spikard-http/src/handler_trait.rs +0 -1015
- data/vendor/crates/spikard-http/src/handler_trait_tests.rs +0 -290
- data/vendor/crates/spikard-http/src/jsonrpc/http_handler.rs +0 -502
- data/vendor/crates/spikard-http/src/jsonrpc/method_registry.rs +0 -648
- data/vendor/crates/spikard-http/src/jsonrpc/mod.rs +0 -58
- data/vendor/crates/spikard-http/src/jsonrpc/protocol.rs +0 -1207
- data/vendor/crates/spikard-http/src/jsonrpc/router.rs +0 -2262
- data/vendor/crates/spikard-http/src/lib.rs +0 -548
- data/vendor/crates/spikard-http/src/lifecycle/adapter.rs +0 -230
- data/vendor/crates/spikard-http/src/lifecycle.rs +0 -1193
- data/vendor/crates/spikard-http/src/middleware/mod.rs +0 -560
- data/vendor/crates/spikard-http/src/middleware/multipart.rs +0 -912
- data/vendor/crates/spikard-http/src/middleware/urlencoded.rs +0 -513
- data/vendor/crates/spikard-http/src/middleware/validation.rs +0 -768
- data/vendor/crates/spikard-http/src/openapi/mod.rs +0 -309
- data/vendor/crates/spikard-http/src/openapi/parameter_extraction.rs +0 -535
- data/vendor/crates/spikard-http/src/openapi/schema_conversion.rs +0 -1363
- data/vendor/crates/spikard-http/src/openapi/spec_generation.rs +0 -667
- data/vendor/crates/spikard-http/src/query_parser.rs +0 -793
- data/vendor/crates/spikard-http/src/response.rs +0 -720
- data/vendor/crates/spikard-http/src/server/fast_router.rs +0 -186
- data/vendor/crates/spikard-http/src/server/grpc_routing.rs +0 -858
- data/vendor/crates/spikard-http/src/server/handler.rs +0 -1661
- data/vendor/crates/spikard-http/src/server/lifecycle_execution.rs +0 -253
- data/vendor/crates/spikard-http/src/server/mod.rs +0 -1649
- data/vendor/crates/spikard-http/src/server/request_extraction.rs +0 -871
- data/vendor/crates/spikard-http/src/server/routing_factory.rs +0 -618
- data/vendor/crates/spikard-http/src/sse.rs +0 -1409
- data/vendor/crates/spikard-http/src/testing/form.rs +0 -52
- data/vendor/crates/spikard-http/src/testing/multipart.rs +0 -64
- data/vendor/crates/spikard-http/src/testing/test_client.rs +0 -787
- data/vendor/crates/spikard-http/src/testing.rs +0 -617
- data/vendor/crates/spikard-http/src/websocket.rs +0 -1477
- data/vendor/crates/spikard-http/tests/auth_integration.rs +0 -645
- data/vendor/crates/spikard-http/tests/background_behavior.rs +0 -832
- data/vendor/crates/spikard-http/tests/common/grpc_helpers.rs +0 -1012
- data/vendor/crates/spikard-http/tests/common/handlers.rs +0 -309
- data/vendor/crates/spikard-http/tests/common/mod.rs +0 -33
- data/vendor/crates/spikard-http/tests/common/test_builders.rs +0 -628
- data/vendor/crates/spikard-http/tests/di_handler_error_responses.rs +0 -162
- data/vendor/crates/spikard-http/tests/di_integration.rs +0 -192
- data/vendor/crates/spikard-http/tests/doc_snippets.rs +0 -5
- data/vendor/crates/spikard-http/tests/grpc_bidirectional_streaming.rs +0 -430
- data/vendor/crates/spikard-http/tests/grpc_client_streaming.rs +0 -738
- data/vendor/crates/spikard-http/tests/grpc_error_handling_test.rs +0 -652
- data/vendor/crates/spikard-http/tests/grpc_integration_test.rs +0 -334
- data/vendor/crates/spikard-http/tests/grpc_metadata_test.rs +0 -532
- data/vendor/crates/spikard-http/tests/grpc_server_integration.rs +0 -495
- data/vendor/crates/spikard-http/tests/grpc_server_streaming.rs +0 -974
- data/vendor/crates/spikard-http/tests/lifecycle_execution.rs +0 -1093
- data/vendor/crates/spikard-http/tests/middleware_stack_integration.rs +0 -389
- data/vendor/crates/spikard-http/tests/multipart_behavior.rs +0 -656
- data/vendor/crates/spikard-http/tests/request_extraction_full.rs +0 -513
- data/vendor/crates/spikard-http/tests/server_auth_middleware_behavior.rs +0 -328
- data/vendor/crates/spikard-http/tests/server_config_builder.rs +0 -314
- data/vendor/crates/spikard-http/tests/server_configured_router_behavior.rs +0 -200
- data/vendor/crates/spikard-http/tests/server_cors_preflight.rs +0 -83
- data/vendor/crates/spikard-http/tests/server_handler_wrappers.rs +0 -464
- data/vendor/crates/spikard-http/tests/server_method_router_additional_behavior.rs +0 -286
- data/vendor/crates/spikard-http/tests/server_method_router_coverage.rs +0 -118
- data/vendor/crates/spikard-http/tests/server_middleware_behavior.rs +0 -99
- data/vendor/crates/spikard-http/tests/server_middleware_branches.rs +0 -204
- data/vendor/crates/spikard-http/tests/server_openapi_jsonrpc_static.rs +0 -421
- data/vendor/crates/spikard-http/tests/server_router_behavior.rs +0 -121
- data/vendor/crates/spikard-http/tests/sse_behavior.rs +0 -620
- data/vendor/crates/spikard-http/tests/sse_full_behavior.rs +0 -584
- data/vendor/crates/spikard-http/tests/sse_handler_behavior.rs +0 -130
- data/vendor/crates/spikard-http/tests/test_client_requests.rs +0 -167
- data/vendor/crates/spikard-http/tests/testing_helpers.rs +0 -87
- data/vendor/crates/spikard-http/tests/testing_module_coverage.rs +0 -155
- data/vendor/crates/spikard-http/tests/urlencoded_content_type.rs +0 -82
- data/vendor/crates/spikard-http/tests/websocket_behavior.rs +0 -663
- data/vendor/crates/spikard-http/tests/websocket_full_behavior.rs +0 -440
- data/vendor/crates/spikard-http/tests/websocket_integration.rs +0 -150
- data/vendor/crates/spikard-rb/Cargo.toml +0 -68
- data/vendor/crates/spikard-rb/build.rs +0 -200
- data/vendor/crates/spikard-rb/src/background.rs +0 -63
- data/vendor/crates/spikard-rb/src/config/mod.rs +0 -5
- data/vendor/crates/spikard-rb/src/config/server_config.rs +0 -401
- data/vendor/crates/spikard-rb/src/conversion.rs +0 -688
- data/vendor/crates/spikard-rb/src/di/builder.rs +0 -100
- data/vendor/crates/spikard-rb/src/di/mod.rs +0 -375
- data/vendor/crates/spikard-rb/src/grpc/handler.rs +0 -834
- data/vendor/crates/spikard-rb/src/grpc/mod.rs +0 -13
- data/vendor/crates/spikard-rb/src/gvl.rs +0 -80
- data/vendor/crates/spikard-rb/src/handler.rs +0 -699
- data/vendor/crates/spikard-rb/src/integration/mod.rs +0 -3
- data/vendor/crates/spikard-rb/src/lib.rs +0 -2264
- data/vendor/crates/spikard-rb/src/lifecycle.rs +0 -303
- data/vendor/crates/spikard-rb/src/metadata/mod.rs +0 -5
- data/vendor/crates/spikard-rb/src/metadata/route_extraction.rs +0 -507
- data/vendor/crates/spikard-rb/src/request.rs +0 -439
- data/vendor/crates/spikard-rb/src/runtime/mod.rs +0 -5
- data/vendor/crates/spikard-rb/src/runtime/server_runner.rs +0 -344
- data/vendor/crates/spikard-rb/src/server.rs +0 -307
- data/vendor/crates/spikard-rb/src/sse.rs +0 -231
- data/vendor/crates/spikard-rb/src/testing/client.rs +0 -698
- data/vendor/crates/spikard-rb/src/testing/mod.rs +0 -7
- data/vendor/crates/spikard-rb/src/testing/sse.rs +0 -108
- data/vendor/crates/spikard-rb/src/testing/websocket.rs +0 -573
- data/vendor/crates/spikard-rb/src/websocket.rs +0 -475
- data/vendor/crates/spikard-rb-macros/Cargo.toml +0 -25
- data/vendor/crates/spikard-rb-macros/src/lib.rs +0 -51
|
@@ -1,72 +0,0 @@
|
|
|
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
|
-
///
|
|
35
|
-
/// # Errors
|
|
36
|
-
/// Returns a structured error if a panic occurs during function execution.
|
|
37
|
-
pub fn shield_panic<T, F>(f: F) -> Result<T, StructuredError>
|
|
38
|
-
where
|
|
39
|
-
F: FnOnce() -> T + UnwindSafe,
|
|
40
|
-
{
|
|
41
|
-
catch_unwind(f).map_err(|_| StructuredError::simple("panic", "Unexpected panic in Rust code"))
|
|
42
|
-
}
|
|
43
|
-
|
|
44
|
-
#[cfg(test)]
|
|
45
|
-
mod tests {
|
|
46
|
-
use super::*;
|
|
47
|
-
use serde_json::json;
|
|
48
|
-
|
|
49
|
-
#[test]
|
|
50
|
-
fn structured_error_constructors_populate_fields() {
|
|
51
|
-
let details = json!({"field": "name"});
|
|
52
|
-
let err = StructuredError::new("invalid", "bad input", details.clone());
|
|
53
|
-
assert_eq!(err.code, "invalid");
|
|
54
|
-
assert_eq!(err.error, "bad input");
|
|
55
|
-
assert_eq!(err.details, details);
|
|
56
|
-
|
|
57
|
-
let simple = StructuredError::simple("missing", "not found");
|
|
58
|
-
assert_eq!(simple.code, "missing");
|
|
59
|
-
assert_eq!(simple.error, "not found");
|
|
60
|
-
assert!(simple.details.is_object());
|
|
61
|
-
}
|
|
62
|
-
|
|
63
|
-
#[test]
|
|
64
|
-
fn shield_panic_returns_ok_or_structured_error() {
|
|
65
|
-
let ok = shield_panic(|| 42);
|
|
66
|
-
assert_eq!(ok.unwrap(), 42);
|
|
67
|
-
|
|
68
|
-
let err = shield_panic(|| panic!("boom")).unwrap_err();
|
|
69
|
-
assert_eq!(err.code, "panic");
|
|
70
|
-
assert!(err.error.contains("Unexpected panic"));
|
|
71
|
-
}
|
|
72
|
-
}
|
|
@@ -1,492 +0,0 @@
|
|
|
1
|
-
use serde::{Deserialize, Serialize};
|
|
2
|
-
use serde_json::Value;
|
|
3
|
-
use std::sync::OnceLock;
|
|
4
|
-
|
|
5
|
-
/// HTTP method
|
|
6
|
-
#[derive(Debug, Clone, PartialEq, Eq, Hash, Serialize, Deserialize)]
|
|
7
|
-
pub enum Method {
|
|
8
|
-
Get,
|
|
9
|
-
Post,
|
|
10
|
-
Put,
|
|
11
|
-
Patch,
|
|
12
|
-
Delete,
|
|
13
|
-
Head,
|
|
14
|
-
Options,
|
|
15
|
-
Trace,
|
|
16
|
-
}
|
|
17
|
-
|
|
18
|
-
impl Method {
|
|
19
|
-
#[must_use]
|
|
20
|
-
pub const fn as_str(&self) -> &'static str {
|
|
21
|
-
match self {
|
|
22
|
-
Self::Get => "GET",
|
|
23
|
-
Self::Post => "POST",
|
|
24
|
-
Self::Put => "PUT",
|
|
25
|
-
Self::Patch => "PATCH",
|
|
26
|
-
Self::Delete => "DELETE",
|
|
27
|
-
Self::Head => "HEAD",
|
|
28
|
-
Self::Options => "OPTIONS",
|
|
29
|
-
Self::Trace => "TRACE",
|
|
30
|
-
}
|
|
31
|
-
}
|
|
32
|
-
}
|
|
33
|
-
|
|
34
|
-
impl std::fmt::Display for Method {
|
|
35
|
-
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
|
36
|
-
write!(f, "{}", self.as_str())
|
|
37
|
-
}
|
|
38
|
-
}
|
|
39
|
-
|
|
40
|
-
impl std::str::FromStr for Method {
|
|
41
|
-
type Err = String;
|
|
42
|
-
|
|
43
|
-
fn from_str(s: &str) -> Result<Self, Self::Err> {
|
|
44
|
-
match s.to_uppercase().as_str() {
|
|
45
|
-
"GET" => Ok(Self::Get),
|
|
46
|
-
"POST" => Ok(Self::Post),
|
|
47
|
-
"PUT" => Ok(Self::Put),
|
|
48
|
-
"PATCH" => Ok(Self::Patch),
|
|
49
|
-
"DELETE" => Ok(Self::Delete),
|
|
50
|
-
"HEAD" => Ok(Self::Head),
|
|
51
|
-
"OPTIONS" => Ok(Self::Options),
|
|
52
|
-
"TRACE" => Ok(Self::Trace),
|
|
53
|
-
_ => Err(format!("Unknown HTTP method: {s}")),
|
|
54
|
-
}
|
|
55
|
-
}
|
|
56
|
-
}
|
|
57
|
-
|
|
58
|
-
/// CORS configuration for a route
|
|
59
|
-
#[derive(Debug, Clone, Serialize, Deserialize)]
|
|
60
|
-
pub struct CorsConfig {
|
|
61
|
-
pub allowed_origins: Vec<String>,
|
|
62
|
-
pub allowed_methods: Vec<String>,
|
|
63
|
-
#[serde(default)]
|
|
64
|
-
pub allowed_headers: Vec<String>,
|
|
65
|
-
#[serde(skip_serializing_if = "Option::is_none")]
|
|
66
|
-
pub expose_headers: Option<Vec<String>>,
|
|
67
|
-
#[serde(skip_serializing_if = "Option::is_none")]
|
|
68
|
-
pub max_age: Option<u32>,
|
|
69
|
-
#[serde(skip_serializing_if = "Option::is_none")]
|
|
70
|
-
pub allow_credentials: Option<bool>,
|
|
71
|
-
|
|
72
|
-
// Optimized caches (lazy-initialized on first use)
|
|
73
|
-
#[serde(skip)]
|
|
74
|
-
#[doc(hidden)]
|
|
75
|
-
pub methods_joined_cache: OnceLock<String>,
|
|
76
|
-
#[serde(skip)]
|
|
77
|
-
#[doc(hidden)]
|
|
78
|
-
pub headers_joined_cache: OnceLock<String>,
|
|
79
|
-
}
|
|
80
|
-
|
|
81
|
-
impl CorsConfig {
|
|
82
|
-
/// Get the cached joined methods string for preflight responses
|
|
83
|
-
pub fn allowed_methods_joined(&self) -> &str {
|
|
84
|
-
self.methods_joined_cache
|
|
85
|
-
.get_or_init(|| self.allowed_methods.join(", "))
|
|
86
|
-
}
|
|
87
|
-
|
|
88
|
-
/// Get the cached joined headers string for preflight responses
|
|
89
|
-
pub fn allowed_headers_joined(&self) -> &str {
|
|
90
|
-
self.headers_joined_cache
|
|
91
|
-
.get_or_init(|| self.allowed_headers.join(", "))
|
|
92
|
-
}
|
|
93
|
-
|
|
94
|
-
/// Check if an origin is allowed (O(1) with wildcard, O(n) for exact match)
|
|
95
|
-
pub fn is_origin_allowed(&self, origin: &str) -> bool {
|
|
96
|
-
if origin.is_empty() {
|
|
97
|
-
return false;
|
|
98
|
-
}
|
|
99
|
-
self.allowed_origins.iter().any(|o| o == "*" || o == origin)
|
|
100
|
-
}
|
|
101
|
-
|
|
102
|
-
/// Check if a method is allowed (O(1) with wildcard, O(n) for exact match)
|
|
103
|
-
pub fn is_method_allowed(&self, method: &str) -> bool {
|
|
104
|
-
self.allowed_methods
|
|
105
|
-
.iter()
|
|
106
|
-
.any(|m| m == "*" || m.eq_ignore_ascii_case(method))
|
|
107
|
-
}
|
|
108
|
-
|
|
109
|
-
/// Check if all requested headers are allowed (O(n) where n = num requested headers)
|
|
110
|
-
pub fn are_headers_allowed(&self, requested: &[&str]) -> bool {
|
|
111
|
-
// Check if wildcard is set
|
|
112
|
-
if self.allowed_headers.iter().any(|h| h == "*") {
|
|
113
|
-
return true;
|
|
114
|
-
}
|
|
115
|
-
|
|
116
|
-
// Check each requested header
|
|
117
|
-
requested.iter().all(|req_header| {
|
|
118
|
-
self.allowed_headers
|
|
119
|
-
.iter()
|
|
120
|
-
.any(|h| h.to_lowercase() == req_header.to_lowercase())
|
|
121
|
-
})
|
|
122
|
-
}
|
|
123
|
-
}
|
|
124
|
-
|
|
125
|
-
impl Default for CorsConfig {
|
|
126
|
-
fn default() -> Self {
|
|
127
|
-
Self {
|
|
128
|
-
allowed_origins: vec!["*".to_string()],
|
|
129
|
-
allowed_methods: vec!["*".to_string()],
|
|
130
|
-
allowed_headers: vec![],
|
|
131
|
-
expose_headers: None,
|
|
132
|
-
max_age: None,
|
|
133
|
-
allow_credentials: None,
|
|
134
|
-
methods_joined_cache: OnceLock::new(),
|
|
135
|
-
headers_joined_cache: OnceLock::new(),
|
|
136
|
-
}
|
|
137
|
-
}
|
|
138
|
-
}
|
|
139
|
-
|
|
140
|
-
/// Route metadata extracted from bindings
|
|
141
|
-
#[derive(Debug, Clone, Serialize, Deserialize)]
|
|
142
|
-
pub struct RouteMetadata {
|
|
143
|
-
pub method: String,
|
|
144
|
-
pub path: String,
|
|
145
|
-
pub handler_name: String,
|
|
146
|
-
pub request_schema: Option<Value>,
|
|
147
|
-
pub response_schema: Option<Value>,
|
|
148
|
-
pub parameter_schema: Option<Value>,
|
|
149
|
-
#[serde(skip_serializing_if = "Option::is_none")]
|
|
150
|
-
pub file_params: Option<Value>,
|
|
151
|
-
#[serde(default)]
|
|
152
|
-
pub is_async: bool,
|
|
153
|
-
pub cors: Option<CorsConfig>,
|
|
154
|
-
/// Name of the body parameter (defaults to "body" if not specified)
|
|
155
|
-
#[serde(skip_serializing_if = "Option::is_none")]
|
|
156
|
-
pub body_param_name: Option<String>,
|
|
157
|
-
/// List of dependency keys this handler requires (for DI)
|
|
158
|
-
#[cfg(feature = "di")]
|
|
159
|
-
#[serde(skip_serializing_if = "Option::is_none")]
|
|
160
|
-
pub handler_dependencies: Option<Vec<String>>,
|
|
161
|
-
/// JSON-RPC method metadata (if this route is exposed as a JSON-RPC method)
|
|
162
|
-
#[serde(skip_serializing_if = "Option::is_none")]
|
|
163
|
-
pub jsonrpc_method: Option<Value>,
|
|
164
|
-
/// Optional static response configuration: `{"status": 200, "body": "OK", "content_type": "text/plain"}`
|
|
165
|
-
/// When present, the handler is replaced by a `StaticResponseHandler` that bypasses the full
|
|
166
|
-
/// middleware pipeline for maximum throughput.
|
|
167
|
-
#[serde(skip_serializing_if = "Option::is_none")]
|
|
168
|
-
pub static_response: Option<Value>,
|
|
169
|
-
}
|
|
170
|
-
|
|
171
|
-
/// Compression configuration shared across runtimes
|
|
172
|
-
#[derive(Debug, Clone, Serialize, Deserialize)]
|
|
173
|
-
pub struct CompressionConfig {
|
|
174
|
-
/// Enable gzip compression
|
|
175
|
-
#[serde(default = "default_true")]
|
|
176
|
-
pub gzip: bool,
|
|
177
|
-
/// Enable brotli compression
|
|
178
|
-
#[serde(default = "default_true")]
|
|
179
|
-
pub brotli: bool,
|
|
180
|
-
/// Minimum response size to compress (bytes)
|
|
181
|
-
#[serde(default = "default_compression_min_size")]
|
|
182
|
-
pub min_size: usize,
|
|
183
|
-
/// Compression quality (0-11 for brotli, 0-9 for gzip)
|
|
184
|
-
#[serde(default = "default_compression_quality")]
|
|
185
|
-
pub quality: u32,
|
|
186
|
-
}
|
|
187
|
-
|
|
188
|
-
const fn default_true() -> bool {
|
|
189
|
-
true
|
|
190
|
-
}
|
|
191
|
-
|
|
192
|
-
const fn default_compression_min_size() -> usize {
|
|
193
|
-
1024
|
|
194
|
-
}
|
|
195
|
-
|
|
196
|
-
const fn default_compression_quality() -> u32 {
|
|
197
|
-
6
|
|
198
|
-
}
|
|
199
|
-
|
|
200
|
-
impl Default for CompressionConfig {
|
|
201
|
-
fn default() -> Self {
|
|
202
|
-
Self {
|
|
203
|
-
gzip: true,
|
|
204
|
-
brotli: true,
|
|
205
|
-
min_size: default_compression_min_size(),
|
|
206
|
-
quality: default_compression_quality(),
|
|
207
|
-
}
|
|
208
|
-
}
|
|
209
|
-
}
|
|
210
|
-
|
|
211
|
-
/// Rate limiting configuration shared across runtimes
|
|
212
|
-
#[derive(Debug, Clone, Serialize, Deserialize)]
|
|
213
|
-
pub struct RateLimitConfig {
|
|
214
|
-
/// Requests per second
|
|
215
|
-
pub per_second: u64,
|
|
216
|
-
/// Burst allowance
|
|
217
|
-
pub burst: u32,
|
|
218
|
-
/// Use IP-based rate limiting
|
|
219
|
-
#[serde(default = "default_true")]
|
|
220
|
-
pub ip_based: bool,
|
|
221
|
-
}
|
|
222
|
-
|
|
223
|
-
impl Default for RateLimitConfig {
|
|
224
|
-
fn default() -> Self {
|
|
225
|
-
Self {
|
|
226
|
-
per_second: 100,
|
|
227
|
-
burst: 200,
|
|
228
|
-
ip_based: true,
|
|
229
|
-
}
|
|
230
|
-
}
|
|
231
|
-
}
|
|
232
|
-
|
|
233
|
-
#[cfg(test)]
|
|
234
|
-
mod tests {
|
|
235
|
-
use super::*;
|
|
236
|
-
use std::str::FromStr;
|
|
237
|
-
|
|
238
|
-
#[test]
|
|
239
|
-
fn test_method_as_str_get() {
|
|
240
|
-
assert_eq!(Method::Get.as_str(), "GET");
|
|
241
|
-
}
|
|
242
|
-
|
|
243
|
-
#[test]
|
|
244
|
-
fn test_method_as_str_post() {
|
|
245
|
-
assert_eq!(Method::Post.as_str(), "POST");
|
|
246
|
-
}
|
|
247
|
-
|
|
248
|
-
#[test]
|
|
249
|
-
fn test_method_as_str_put() {
|
|
250
|
-
assert_eq!(Method::Put.as_str(), "PUT");
|
|
251
|
-
}
|
|
252
|
-
|
|
253
|
-
#[test]
|
|
254
|
-
fn test_method_as_str_patch() {
|
|
255
|
-
assert_eq!(Method::Patch.as_str(), "PATCH");
|
|
256
|
-
}
|
|
257
|
-
|
|
258
|
-
#[test]
|
|
259
|
-
fn test_method_as_str_delete() {
|
|
260
|
-
assert_eq!(Method::Delete.as_str(), "DELETE");
|
|
261
|
-
}
|
|
262
|
-
|
|
263
|
-
#[test]
|
|
264
|
-
fn test_method_as_str_head() {
|
|
265
|
-
assert_eq!(Method::Head.as_str(), "HEAD");
|
|
266
|
-
}
|
|
267
|
-
|
|
268
|
-
#[test]
|
|
269
|
-
fn test_method_as_str_options() {
|
|
270
|
-
assert_eq!(Method::Options.as_str(), "OPTIONS");
|
|
271
|
-
}
|
|
272
|
-
|
|
273
|
-
#[test]
|
|
274
|
-
fn test_method_as_str_trace() {
|
|
275
|
-
assert_eq!(Method::Trace.as_str(), "TRACE");
|
|
276
|
-
}
|
|
277
|
-
|
|
278
|
-
#[test]
|
|
279
|
-
fn test_method_display_get() {
|
|
280
|
-
assert_eq!(Method::Get.to_string(), "GET");
|
|
281
|
-
}
|
|
282
|
-
|
|
283
|
-
#[test]
|
|
284
|
-
fn test_method_display_post() {
|
|
285
|
-
assert_eq!(Method::Post.to_string(), "POST");
|
|
286
|
-
}
|
|
287
|
-
|
|
288
|
-
#[test]
|
|
289
|
-
fn test_method_display_put() {
|
|
290
|
-
assert_eq!(Method::Put.to_string(), "PUT");
|
|
291
|
-
}
|
|
292
|
-
|
|
293
|
-
#[test]
|
|
294
|
-
fn test_method_display_patch() {
|
|
295
|
-
assert_eq!(Method::Patch.to_string(), "PATCH");
|
|
296
|
-
}
|
|
297
|
-
|
|
298
|
-
#[test]
|
|
299
|
-
fn test_method_display_delete() {
|
|
300
|
-
assert_eq!(Method::Delete.to_string(), "DELETE");
|
|
301
|
-
}
|
|
302
|
-
|
|
303
|
-
#[test]
|
|
304
|
-
fn test_method_display_head() {
|
|
305
|
-
assert_eq!(Method::Head.to_string(), "HEAD");
|
|
306
|
-
}
|
|
307
|
-
|
|
308
|
-
#[test]
|
|
309
|
-
fn test_method_display_options() {
|
|
310
|
-
assert_eq!(Method::Options.to_string(), "OPTIONS");
|
|
311
|
-
}
|
|
312
|
-
|
|
313
|
-
#[test]
|
|
314
|
-
fn test_method_display_trace() {
|
|
315
|
-
assert_eq!(Method::Trace.to_string(), "TRACE");
|
|
316
|
-
}
|
|
317
|
-
|
|
318
|
-
#[test]
|
|
319
|
-
fn test_from_str_get() {
|
|
320
|
-
assert_eq!(Method::from_str("GET"), Ok(Method::Get));
|
|
321
|
-
}
|
|
322
|
-
|
|
323
|
-
#[test]
|
|
324
|
-
fn test_from_str_post() {
|
|
325
|
-
assert_eq!(Method::from_str("POST"), Ok(Method::Post));
|
|
326
|
-
}
|
|
327
|
-
|
|
328
|
-
#[test]
|
|
329
|
-
fn test_from_str_put() {
|
|
330
|
-
assert_eq!(Method::from_str("PUT"), Ok(Method::Put));
|
|
331
|
-
}
|
|
332
|
-
|
|
333
|
-
#[test]
|
|
334
|
-
fn test_from_str_patch() {
|
|
335
|
-
assert_eq!(Method::from_str("PATCH"), Ok(Method::Patch));
|
|
336
|
-
}
|
|
337
|
-
|
|
338
|
-
#[test]
|
|
339
|
-
fn test_from_str_delete() {
|
|
340
|
-
assert_eq!(Method::from_str("DELETE"), Ok(Method::Delete));
|
|
341
|
-
}
|
|
342
|
-
|
|
343
|
-
#[test]
|
|
344
|
-
fn test_from_str_head() {
|
|
345
|
-
assert_eq!(Method::from_str("HEAD"), Ok(Method::Head));
|
|
346
|
-
}
|
|
347
|
-
|
|
348
|
-
#[test]
|
|
349
|
-
fn test_from_str_options() {
|
|
350
|
-
assert_eq!(Method::from_str("OPTIONS"), Ok(Method::Options));
|
|
351
|
-
}
|
|
352
|
-
|
|
353
|
-
#[test]
|
|
354
|
-
fn test_from_str_trace() {
|
|
355
|
-
assert_eq!(Method::from_str("TRACE"), Ok(Method::Trace));
|
|
356
|
-
}
|
|
357
|
-
|
|
358
|
-
#[test]
|
|
359
|
-
fn test_from_str_lowercase() {
|
|
360
|
-
assert_eq!(Method::from_str("get"), Ok(Method::Get));
|
|
361
|
-
}
|
|
362
|
-
|
|
363
|
-
#[test]
|
|
364
|
-
fn test_from_str_mixed_case() {
|
|
365
|
-
assert_eq!(Method::from_str("PoSt"), Ok(Method::Post));
|
|
366
|
-
}
|
|
367
|
-
|
|
368
|
-
#[test]
|
|
369
|
-
fn test_from_str_invalid_method() {
|
|
370
|
-
let result = Method::from_str("INVALID");
|
|
371
|
-
assert!(result.is_err());
|
|
372
|
-
assert_eq!(result.unwrap_err(), "Unknown HTTP method: INVALID");
|
|
373
|
-
}
|
|
374
|
-
|
|
375
|
-
#[test]
|
|
376
|
-
fn test_from_str_empty_string() {
|
|
377
|
-
let result = Method::from_str("");
|
|
378
|
-
assert!(result.is_err());
|
|
379
|
-
assert_eq!(result.unwrap_err(), "Unknown HTTP method: ");
|
|
380
|
-
}
|
|
381
|
-
|
|
382
|
-
#[test]
|
|
383
|
-
fn test_compression_config_default() {
|
|
384
|
-
let config = CompressionConfig::default();
|
|
385
|
-
assert!(config.gzip);
|
|
386
|
-
assert!(config.brotli);
|
|
387
|
-
assert_eq!(config.min_size, 1024);
|
|
388
|
-
assert_eq!(config.quality, 6);
|
|
389
|
-
}
|
|
390
|
-
|
|
391
|
-
#[test]
|
|
392
|
-
fn test_default_true() {
|
|
393
|
-
assert!(default_true());
|
|
394
|
-
}
|
|
395
|
-
|
|
396
|
-
#[test]
|
|
397
|
-
fn test_default_compression_min_size() {
|
|
398
|
-
assert_eq!(default_compression_min_size(), 1024);
|
|
399
|
-
}
|
|
400
|
-
|
|
401
|
-
#[test]
|
|
402
|
-
fn test_default_compression_quality() {
|
|
403
|
-
assert_eq!(default_compression_quality(), 6);
|
|
404
|
-
}
|
|
405
|
-
|
|
406
|
-
#[test]
|
|
407
|
-
fn test_rate_limit_config_default() {
|
|
408
|
-
let config = RateLimitConfig::default();
|
|
409
|
-
assert_eq!(config.per_second, 100);
|
|
410
|
-
assert_eq!(config.burst, 200);
|
|
411
|
-
assert!(config.ip_based);
|
|
412
|
-
}
|
|
413
|
-
|
|
414
|
-
#[test]
|
|
415
|
-
fn test_method_equality() {
|
|
416
|
-
assert_eq!(Method::Get, Method::Get);
|
|
417
|
-
assert_ne!(Method::Get, Method::Post);
|
|
418
|
-
}
|
|
419
|
-
|
|
420
|
-
#[test]
|
|
421
|
-
fn test_method_clone() {
|
|
422
|
-
let method = Method::Post;
|
|
423
|
-
let cloned = method.clone();
|
|
424
|
-
assert_eq!(method, cloned);
|
|
425
|
-
}
|
|
426
|
-
|
|
427
|
-
#[test]
|
|
428
|
-
fn test_compression_config_custom_values() {
|
|
429
|
-
let config = CompressionConfig {
|
|
430
|
-
gzip: false,
|
|
431
|
-
brotli: false,
|
|
432
|
-
min_size: 2048,
|
|
433
|
-
quality: 11,
|
|
434
|
-
};
|
|
435
|
-
assert!(!config.gzip);
|
|
436
|
-
assert!(!config.brotli);
|
|
437
|
-
assert_eq!(config.min_size, 2048);
|
|
438
|
-
assert_eq!(config.quality, 11);
|
|
439
|
-
}
|
|
440
|
-
|
|
441
|
-
#[test]
|
|
442
|
-
fn test_rate_limit_config_custom_values() {
|
|
443
|
-
let config = RateLimitConfig {
|
|
444
|
-
per_second: 50,
|
|
445
|
-
burst: 100,
|
|
446
|
-
ip_based: false,
|
|
447
|
-
};
|
|
448
|
-
assert_eq!(config.per_second, 50);
|
|
449
|
-
assert_eq!(config.burst, 100);
|
|
450
|
-
assert!(!config.ip_based);
|
|
451
|
-
}
|
|
452
|
-
|
|
453
|
-
#[test]
|
|
454
|
-
fn test_cors_config_construction() {
|
|
455
|
-
let cors = CorsConfig {
|
|
456
|
-
allowed_origins: vec!["http://localhost:3000".to_string()],
|
|
457
|
-
allowed_methods: vec!["GET".to_string(), "POST".to_string()],
|
|
458
|
-
allowed_headers: vec![],
|
|
459
|
-
expose_headers: None,
|
|
460
|
-
max_age: None,
|
|
461
|
-
allow_credentials: None,
|
|
462
|
-
..Default::default()
|
|
463
|
-
};
|
|
464
|
-
assert_eq!(cors.allowed_origins.len(), 1);
|
|
465
|
-
assert_eq!(cors.allowed_methods.len(), 2);
|
|
466
|
-
assert_eq!(cors.allowed_headers.len(), 0);
|
|
467
|
-
}
|
|
468
|
-
|
|
469
|
-
#[test]
|
|
470
|
-
fn test_route_metadata_construction() {
|
|
471
|
-
let metadata = RouteMetadata {
|
|
472
|
-
method: "GET".to_string(),
|
|
473
|
-
path: "/api/users".to_string(),
|
|
474
|
-
handler_name: "get_users".to_string(),
|
|
475
|
-
request_schema: None,
|
|
476
|
-
response_schema: None,
|
|
477
|
-
parameter_schema: None,
|
|
478
|
-
file_params: None,
|
|
479
|
-
is_async: true,
|
|
480
|
-
cors: None,
|
|
481
|
-
body_param_name: None,
|
|
482
|
-
#[cfg(feature = "di")]
|
|
483
|
-
handler_dependencies: None,
|
|
484
|
-
jsonrpc_method: None,
|
|
485
|
-
static_response: None,
|
|
486
|
-
};
|
|
487
|
-
assert_eq!(metadata.method, "GET");
|
|
488
|
-
assert_eq!(metadata.path, "/api/users");
|
|
489
|
-
assert_eq!(metadata.handler_name, "get_users");
|
|
490
|
-
assert!(metadata.is_async);
|
|
491
|
-
}
|
|
492
|
-
}
|
|
@@ -1,29 +0,0 @@
|
|
|
1
|
-
pub mod bindings;
|
|
2
|
-
pub mod debug;
|
|
3
|
-
#[cfg(feature = "di")]
|
|
4
|
-
pub mod di;
|
|
5
|
-
pub mod errors;
|
|
6
|
-
pub mod http;
|
|
7
|
-
pub mod lifecycle;
|
|
8
|
-
pub mod parameters;
|
|
9
|
-
pub mod problem;
|
|
10
|
-
pub mod request_data;
|
|
11
|
-
pub mod router;
|
|
12
|
-
pub mod schema_registry;
|
|
13
|
-
pub mod type_hints;
|
|
14
|
-
pub mod validation;
|
|
15
|
-
|
|
16
|
-
pub use bindings::response::{RawResponse, StaticAsset};
|
|
17
|
-
#[cfg(feature = "di")]
|
|
18
|
-
pub use di::{
|
|
19
|
-
Dependency, DependencyContainer, DependencyError, DependencyGraph, FactoryDependency, FactoryDependencyBuilder,
|
|
20
|
-
ResolvedDependencies, ValueDependency,
|
|
21
|
-
};
|
|
22
|
-
pub use http::{CompressionConfig, CorsConfig, Method, RateLimitConfig, RouteMetadata};
|
|
23
|
-
pub use lifecycle::{HookResult, LifecycleHook, LifecycleHooks, LifecycleHooksBuilder, request_hook, response_hook};
|
|
24
|
-
pub use parameters::ParameterValidator;
|
|
25
|
-
pub use problem::ProblemDetails;
|
|
26
|
-
pub use request_data::RequestData;
|
|
27
|
-
pub use router::{JsonRpcMethodInfo, Route, RouteHandler, Router};
|
|
28
|
-
pub use schema_registry::SchemaRegistry;
|
|
29
|
-
pub use validation::{SchemaValidator, ValidationError, ValidationErrorDetail};
|