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.
Files changed (142) hide show
  1. checksums.yaml +4 -4
  2. data/LICENSE +1 -1
  3. data/README.md +674 -659
  4. data/ext/spikard_rb/Cargo.toml +17 -17
  5. data/ext/spikard_rb/extconf.rb +10 -10
  6. data/ext/spikard_rb/src/lib.rs +6 -6
  7. data/lib/spikard/app.rb +405 -386
  8. data/lib/spikard/background.rb +27 -27
  9. data/lib/spikard/config.rb +396 -396
  10. data/lib/spikard/converters.rb +13 -13
  11. data/lib/spikard/handler_wrapper.rb +113 -113
  12. data/lib/spikard/provide.rb +214 -214
  13. data/lib/spikard/response.rb +173 -173
  14. data/lib/spikard/schema.rb +243 -243
  15. data/lib/spikard/sse.rb +111 -111
  16. data/lib/spikard/streaming_response.rb +44 -44
  17. data/lib/spikard/testing.rb +256 -221
  18. data/lib/spikard/upload_file.rb +131 -131
  19. data/lib/spikard/version.rb +5 -5
  20. data/lib/spikard/websocket.rb +59 -59
  21. data/lib/spikard.rb +43 -43
  22. data/sig/spikard.rbs +366 -360
  23. data/vendor/crates/spikard-bindings-shared/Cargo.toml +63 -0
  24. data/vendor/crates/spikard-bindings-shared/examples/config_extraction.rs +132 -0
  25. data/vendor/crates/spikard-bindings-shared/src/config_extractor.rs +752 -0
  26. data/vendor/crates/spikard-bindings-shared/src/conversion_traits.rs +194 -0
  27. data/vendor/crates/spikard-bindings-shared/src/di_traits.rs +246 -0
  28. data/vendor/crates/spikard-bindings-shared/src/error_response.rs +401 -0
  29. data/vendor/crates/spikard-bindings-shared/src/handler_base.rs +238 -0
  30. data/vendor/crates/spikard-bindings-shared/src/lib.rs +24 -0
  31. data/vendor/crates/spikard-bindings-shared/src/lifecycle_base.rs +292 -0
  32. data/vendor/crates/spikard-bindings-shared/src/lifecycle_executor.rs +616 -0
  33. data/vendor/crates/spikard-bindings-shared/src/response_builder.rs +305 -0
  34. data/vendor/crates/spikard-bindings-shared/src/test_client_base.rs +248 -0
  35. data/vendor/crates/spikard-bindings-shared/src/validation_helpers.rs +351 -0
  36. data/vendor/crates/spikard-bindings-shared/tests/comprehensive_coverage.rs +454 -0
  37. data/vendor/crates/spikard-bindings-shared/tests/error_response_edge_cases.rs +383 -0
  38. data/vendor/crates/spikard-bindings-shared/tests/handler_base_integration.rs +280 -0
  39. data/vendor/crates/spikard-core/Cargo.toml +40 -40
  40. data/vendor/crates/spikard-core/src/bindings/mod.rs +3 -3
  41. data/vendor/crates/spikard-core/src/bindings/response.rs +133 -133
  42. data/vendor/crates/spikard-core/src/debug.rs +127 -63
  43. data/vendor/crates/spikard-core/src/di/container.rs +702 -726
  44. data/vendor/crates/spikard-core/src/di/dependency.rs +273 -273
  45. data/vendor/crates/spikard-core/src/di/error.rs +118 -118
  46. data/vendor/crates/spikard-core/src/di/factory.rs +534 -538
  47. data/vendor/crates/spikard-core/src/di/graph.rs +506 -545
  48. data/vendor/crates/spikard-core/src/di/mod.rs +192 -192
  49. data/vendor/crates/spikard-core/src/di/resolved.rs +405 -411
  50. data/vendor/crates/spikard-core/src/di/value.rs +281 -283
  51. data/vendor/crates/spikard-core/src/errors.rs +69 -39
  52. data/vendor/crates/spikard-core/src/http.rs +415 -153
  53. data/vendor/crates/spikard-core/src/lib.rs +29 -29
  54. data/vendor/crates/spikard-core/src/lifecycle.rs +1186 -422
  55. data/vendor/crates/spikard-core/src/metadata.rs +389 -0
  56. data/vendor/crates/spikard-core/src/parameters.rs +2525 -722
  57. data/vendor/crates/spikard-core/src/problem.rs +344 -310
  58. data/vendor/crates/spikard-core/src/request_data.rs +1154 -189
  59. data/vendor/crates/spikard-core/src/router.rs +510 -249
  60. data/vendor/crates/spikard-core/src/schema_registry.rs +183 -183
  61. data/vendor/crates/spikard-core/src/type_hints.rs +304 -304
  62. data/vendor/crates/spikard-core/src/validation/error_mapper.rs +688 -0
  63. data/vendor/crates/spikard-core/src/{validation.rs → validation/mod.rs} +457 -699
  64. data/vendor/crates/spikard-http/Cargo.toml +64 -68
  65. data/vendor/crates/spikard-http/examples/sse-notifications.rs +148 -0
  66. data/vendor/crates/spikard-http/examples/websocket-chat.rs +92 -0
  67. data/vendor/crates/spikard-http/src/auth.rs +296 -247
  68. data/vendor/crates/spikard-http/src/background.rs +1860 -249
  69. data/vendor/crates/spikard-http/src/bindings/mod.rs +3 -3
  70. data/vendor/crates/spikard-http/src/bindings/response.rs +1 -1
  71. data/vendor/crates/spikard-http/src/body_metadata.rs +8 -8
  72. data/vendor/crates/spikard-http/src/cors.rs +1005 -490
  73. data/vendor/crates/spikard-http/src/debug.rs +128 -63
  74. data/vendor/crates/spikard-http/src/di_handler.rs +1668 -423
  75. data/vendor/crates/spikard-http/src/handler_response.rs +901 -190
  76. data/vendor/crates/spikard-http/src/handler_trait.rs +830 -228
  77. data/vendor/crates/spikard-http/src/handler_trait_tests.rs +290 -284
  78. data/vendor/crates/spikard-http/src/lib.rs +534 -529
  79. data/vendor/crates/spikard-http/src/lifecycle/adapter.rs +230 -149
  80. data/vendor/crates/spikard-http/src/lifecycle.rs +1193 -428
  81. data/vendor/crates/spikard-http/src/middleware/mod.rs +540 -285
  82. data/vendor/crates/spikard-http/src/middleware/multipart.rs +912 -86
  83. data/vendor/crates/spikard-http/src/middleware/urlencoded.rs +513 -147
  84. data/vendor/crates/spikard-http/src/middleware/validation.rs +735 -287
  85. data/vendor/crates/spikard-http/src/openapi/mod.rs +309 -309
  86. data/vendor/crates/spikard-http/src/openapi/parameter_extraction.rs +535 -190
  87. data/vendor/crates/spikard-http/src/openapi/schema_conversion.rs +1363 -308
  88. data/vendor/crates/spikard-http/src/openapi/spec_generation.rs +665 -195
  89. data/vendor/crates/spikard-http/src/query_parser.rs +793 -369
  90. data/vendor/crates/spikard-http/src/response.rs +720 -399
  91. data/vendor/crates/spikard-http/src/server/handler.rs +1650 -87
  92. data/vendor/crates/spikard-http/src/server/lifecycle_execution.rs +234 -98
  93. data/vendor/crates/spikard-http/src/server/mod.rs +1502 -805
  94. data/vendor/crates/spikard-http/src/server/request_extraction.rs +770 -119
  95. data/vendor/crates/spikard-http/src/server/routing_factory.rs +599 -0
  96. data/vendor/crates/spikard-http/src/sse.rs +1409 -447
  97. data/vendor/crates/spikard-http/src/testing/form.rs +52 -14
  98. data/vendor/crates/spikard-http/src/testing/multipart.rs +60 -60
  99. data/vendor/crates/spikard-http/src/testing/test_client.rs +283 -285
  100. data/vendor/crates/spikard-http/src/testing.rs +377 -377
  101. data/vendor/crates/spikard-http/src/websocket.rs +1375 -324
  102. data/vendor/crates/spikard-http/tests/background_behavior.rs +832 -0
  103. data/vendor/crates/spikard-http/tests/common/handlers.rs +309 -0
  104. data/vendor/crates/spikard-http/tests/common/mod.rs +26 -0
  105. data/vendor/crates/spikard-http/tests/di_integration.rs +192 -0
  106. data/vendor/crates/spikard-http/tests/doc_snippets.rs +5 -0
  107. data/vendor/crates/spikard-http/tests/lifecycle_execution.rs +1093 -0
  108. data/vendor/crates/spikard-http/tests/multipart_behavior.rs +656 -0
  109. data/vendor/crates/spikard-http/tests/server_config_builder.rs +314 -0
  110. data/vendor/crates/spikard-http/tests/sse_behavior.rs +620 -0
  111. data/vendor/crates/spikard-http/tests/websocket_behavior.rs +663 -0
  112. data/vendor/crates/spikard-rb/Cargo.toml +48 -42
  113. data/vendor/crates/spikard-rb/build.rs +199 -8
  114. data/vendor/crates/spikard-rb/src/background.rs +63 -63
  115. data/vendor/crates/spikard-rb/src/config/mod.rs +5 -0
  116. data/vendor/crates/spikard-rb/src/{config.rs → config/server_config.rs} +285 -294
  117. data/vendor/crates/spikard-rb/src/conversion.rs +554 -453
  118. data/vendor/crates/spikard-rb/src/di/builder.rs +100 -0
  119. data/vendor/crates/spikard-rb/src/{di.rs → di/mod.rs} +375 -409
  120. data/vendor/crates/spikard-rb/src/handler.rs +618 -625
  121. data/vendor/crates/spikard-rb/src/integration/mod.rs +3 -0
  122. data/vendor/crates/spikard-rb/src/lib.rs +1810 -2771
  123. data/vendor/crates/spikard-rb/src/lifecycle.rs +275 -274
  124. data/vendor/crates/spikard-rb/src/metadata/mod.rs +5 -0
  125. data/vendor/crates/spikard-rb/src/metadata/route_extraction.rs +447 -0
  126. data/vendor/crates/spikard-rb/src/runtime/mod.rs +5 -0
  127. data/vendor/crates/spikard-rb/src/runtime/server_runner.rs +324 -0
  128. data/vendor/crates/spikard-rb/src/server.rs +308 -283
  129. data/vendor/crates/spikard-rb/src/sse.rs +231 -231
  130. data/vendor/crates/spikard-rb/src/{test_client.rs → testing/client.rs} +551 -404
  131. data/vendor/crates/spikard-rb/src/testing/mod.rs +7 -0
  132. data/vendor/crates/spikard-rb/src/{test_sse.rs → testing/sse.rs} +143 -143
  133. data/vendor/crates/spikard-rb/src/testing/websocket.rs +635 -0
  134. data/vendor/crates/spikard-rb/src/websocket.rs +374 -233
  135. metadata +46 -13
  136. data/vendor/crates/spikard-http/src/parameters.rs +0 -1
  137. data/vendor/crates/spikard-http/src/problem.rs +0 -1
  138. data/vendor/crates/spikard-http/src/router.rs +0 -1
  139. data/vendor/crates/spikard-http/src/schema_registry.rs +0 -1
  140. data/vendor/crates/spikard-http/src/type_hints.rs +0 -1
  141. data/vendor/crates/spikard-http/src/validation.rs +0 -1
  142. 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
- /// Compression configuration shared across runtimes
94
- #[derive(Debug, Clone, Serialize, Deserialize)]
95
- pub struct CompressionConfig {
96
- /// Enable gzip compression
97
- #[serde(default = "default_true")]
98
- pub gzip: bool,
99
- /// Enable brotli compression
100
- #[serde(default = "default_true")]
101
- pub brotli: bool,
102
- /// Minimum response size to compress (bytes)
103
- #[serde(default = "default_compression_min_size")]
104
- pub min_size: usize,
105
- /// Compression quality (0-11 for brotli, 0-9 for gzip)
106
- #[serde(default = "default_compression_quality")]
107
- pub quality: u32,
108
- }
109
-
110
- const fn default_true() -> bool {
111
- true
112
- }
113
-
114
- const fn default_compression_min_size() -> usize {
115
- 1024
116
- }
117
-
118
- const fn default_compression_quality() -> u32 {
119
- 6
120
- }
121
-
122
- impl Default for CompressionConfig {
123
- fn default() -> Self {
124
- Self {
125
- gzip: true,
126
- brotli: true,
127
- min_size: default_compression_min_size(),
128
- quality: default_compression_quality(),
129
- }
130
- }
131
- }
132
-
133
- /// Rate limiting configuration shared across runtimes
134
- #[derive(Debug, Clone, Serialize, Deserialize)]
135
- pub struct RateLimitConfig {
136
- /// Requests per second
137
- pub per_second: u64,
138
- /// Burst allowance
139
- pub burst: u32,
140
- /// Use IP-based rate limiting
141
- #[serde(default = "default_true")]
142
- pub ip_based: bool,
143
- }
144
-
145
- impl Default for RateLimitConfig {
146
- fn default() -> Self {
147
- Self {
148
- per_second: 100,
149
- burst: 200,
150
- ip_based: true,
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
+ }