spikard 0.3.2 → 0.3.3
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 +659 -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 +386 -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 +221 -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 +360 -360
- 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 +63 -63
- data/vendor/crates/spikard-core/src/di/container.rs +726 -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 +538 -538
- data/vendor/crates/spikard-core/src/di/graph.rs +545 -545
- data/vendor/crates/spikard-core/src/di/mod.rs +192 -192
- data/vendor/crates/spikard-core/src/di/resolved.rs +411 -411
- data/vendor/crates/spikard-core/src/di/value.rs +283 -283
- data/vendor/crates/spikard-core/src/errors.rs +39 -39
- data/vendor/crates/spikard-core/src/http.rs +153 -153
- data/vendor/crates/spikard-core/src/lib.rs +29 -29
- data/vendor/crates/spikard-core/src/lifecycle.rs +422 -422
- data/vendor/crates/spikard-core/src/parameters.rs +722 -722
- data/vendor/crates/spikard-core/src/problem.rs +310 -310
- data/vendor/crates/spikard-core/src/request_data.rs +189 -189
- data/vendor/crates/spikard-core/src/router.rs +249 -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.rs +699 -699
- data/vendor/crates/spikard-http/Cargo.toml +58 -58
- data/vendor/crates/spikard-http/src/auth.rs +247 -247
- data/vendor/crates/spikard-http/src/background.rs +249 -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 +490 -490
- data/vendor/crates/spikard-http/src/debug.rs +63 -63
- data/vendor/crates/spikard-http/src/di_handler.rs +423 -423
- data/vendor/crates/spikard-http/src/handler_response.rs +190 -190
- data/vendor/crates/spikard-http/src/handler_trait.rs +228 -228
- data/vendor/crates/spikard-http/src/handler_trait_tests.rs +284 -284
- data/vendor/crates/spikard-http/src/lib.rs +529 -529
- data/vendor/crates/spikard-http/src/lifecycle/adapter.rs +149 -149
- data/vendor/crates/spikard-http/src/lifecycle.rs +428 -428
- data/vendor/crates/spikard-http/src/middleware/mod.rs +285 -285
- data/vendor/crates/spikard-http/src/middleware/multipart.rs +86 -86
- data/vendor/crates/spikard-http/src/middleware/urlencoded.rs +147 -147
- data/vendor/crates/spikard-http/src/middleware/validation.rs +287 -287
- data/vendor/crates/spikard-http/src/openapi/mod.rs +309 -309
- data/vendor/crates/spikard-http/src/openapi/parameter_extraction.rs +190 -190
- data/vendor/crates/spikard-http/src/openapi/schema_conversion.rs +308 -308
- data/vendor/crates/spikard-http/src/openapi/spec_generation.rs +195 -195
- data/vendor/crates/spikard-http/src/parameters.rs +1 -1
- data/vendor/crates/spikard-http/src/problem.rs +1 -1
- data/vendor/crates/spikard-http/src/query_parser.rs +369 -369
- data/vendor/crates/spikard-http/src/response.rs +399 -399
- data/vendor/crates/spikard-http/src/router.rs +1 -1
- data/vendor/crates/spikard-http/src/schema_registry.rs +1 -1
- data/vendor/crates/spikard-http/src/server/handler.rs +87 -87
- data/vendor/crates/spikard-http/src/server/lifecycle_execution.rs +98 -98
- data/vendor/crates/spikard-http/src/server/mod.rs +805 -805
- data/vendor/crates/spikard-http/src/server/request_extraction.rs +119 -119
- data/vendor/crates/spikard-http/src/sse.rs +447 -447
- data/vendor/crates/spikard-http/src/testing/form.rs +14 -14
- data/vendor/crates/spikard-http/src/testing/multipart.rs +60 -60
- data/vendor/crates/spikard-http/src/testing/test_client.rs +285 -285
- data/vendor/crates/spikard-http/src/testing.rs +377 -377
- data/vendor/crates/spikard-http/src/type_hints.rs +1 -1
- data/vendor/crates/spikard-http/src/validation.rs +1 -1
- data/vendor/crates/spikard-http/src/websocket.rs +324 -324
- data/vendor/crates/spikard-rb/Cargo.toml +42 -42
- data/vendor/crates/spikard-rb/build.rs +8 -8
- data/vendor/crates/spikard-rb/src/background.rs +63 -63
- data/vendor/crates/spikard-rb/src/config.rs +294 -294
- data/vendor/crates/spikard-rb/src/conversion.rs +453 -453
- data/vendor/crates/spikard-rb/src/di.rs +409 -409
- data/vendor/crates/spikard-rb/src/handler.rs +625 -625
- data/vendor/crates/spikard-rb/src/lib.rs +2771 -2771
- data/vendor/crates/spikard-rb/src/lifecycle.rs +274 -274
- data/vendor/crates/spikard-rb/src/server.rs +283 -283
- data/vendor/crates/spikard-rb/src/sse.rs +231 -231
- data/vendor/crates/spikard-rb/src/test_client.rs +404 -404
- data/vendor/crates/spikard-rb/src/test_sse.rs +143 -143
- data/vendor/crates/spikard-rb/src/test_websocket.rs +221 -221
- data/vendor/crates/spikard-rb/src/websocket.rs +233 -233
- data/vendor/spikard-core/Cargo.toml +40 -40
- data/vendor/spikard-core/src/bindings/mod.rs +3 -3
- data/vendor/spikard-core/src/bindings/response.rs +133 -133
- data/vendor/spikard-core/src/debug.rs +63 -63
- data/vendor/spikard-core/src/di/container.rs +726 -726
- data/vendor/spikard-core/src/di/dependency.rs +273 -273
- data/vendor/spikard-core/src/di/error.rs +118 -118
- data/vendor/spikard-core/src/di/factory.rs +538 -538
- data/vendor/spikard-core/src/di/graph.rs +545 -545
- data/vendor/spikard-core/src/di/mod.rs +192 -192
- data/vendor/spikard-core/src/di/resolved.rs +411 -411
- data/vendor/spikard-core/src/di/value.rs +283 -283
- data/vendor/spikard-core/src/http.rs +153 -153
- data/vendor/spikard-core/src/lib.rs +28 -28
- data/vendor/spikard-core/src/lifecycle.rs +422 -422
- data/vendor/spikard-core/src/parameters.rs +719 -719
- data/vendor/spikard-core/src/problem.rs +310 -310
- data/vendor/spikard-core/src/request_data.rs +189 -189
- data/vendor/spikard-core/src/router.rs +249 -249
- data/vendor/spikard-core/src/schema_registry.rs +183 -183
- data/vendor/spikard-core/src/type_hints.rs +304 -304
- data/vendor/spikard-core/src/validation.rs +699 -699
- data/vendor/spikard-http/Cargo.toml +58 -58
- data/vendor/spikard-http/src/auth.rs +247 -247
- data/vendor/spikard-http/src/background.rs +249 -249
- data/vendor/spikard-http/src/bindings/mod.rs +3 -3
- data/vendor/spikard-http/src/bindings/response.rs +1 -1
- data/vendor/spikard-http/src/body_metadata.rs +8 -8
- data/vendor/spikard-http/src/cors.rs +490 -490
- data/vendor/spikard-http/src/debug.rs +63 -63
- data/vendor/spikard-http/src/di_handler.rs +423 -423
- data/vendor/spikard-http/src/handler_response.rs +190 -190
- data/vendor/spikard-http/src/handler_trait.rs +228 -228
- data/vendor/spikard-http/src/handler_trait_tests.rs +284 -284
- data/vendor/spikard-http/src/lib.rs +529 -529
- data/vendor/spikard-http/src/lifecycle/adapter.rs +149 -149
- data/vendor/spikard-http/src/lifecycle.rs +428 -428
- data/vendor/spikard-http/src/middleware/mod.rs +285 -285
- data/vendor/spikard-http/src/middleware/multipart.rs +86 -86
- data/vendor/spikard-http/src/middleware/urlencoded.rs +147 -147
- data/vendor/spikard-http/src/middleware/validation.rs +287 -287
- data/vendor/spikard-http/src/openapi/mod.rs +309 -309
- data/vendor/spikard-http/src/openapi/parameter_extraction.rs +190 -190
- data/vendor/spikard-http/src/openapi/schema_conversion.rs +308 -308
- data/vendor/spikard-http/src/openapi/spec_generation.rs +195 -195
- data/vendor/spikard-http/src/parameters.rs +1 -1
- data/vendor/spikard-http/src/problem.rs +1 -1
- data/vendor/spikard-http/src/query_parser.rs +369 -369
- data/vendor/spikard-http/src/response.rs +399 -399
- data/vendor/spikard-http/src/router.rs +1 -1
- data/vendor/spikard-http/src/schema_registry.rs +1 -1
- data/vendor/spikard-http/src/server/handler.rs +80 -80
- data/vendor/spikard-http/src/server/lifecycle_execution.rs +98 -98
- data/vendor/spikard-http/src/server/mod.rs +805 -805
- data/vendor/spikard-http/src/server/request_extraction.rs +119 -119
- data/vendor/spikard-http/src/sse.rs +447 -447
- data/vendor/spikard-http/src/testing/form.rs +14 -14
- data/vendor/spikard-http/src/testing/multipart.rs +60 -60
- data/vendor/spikard-http/src/testing/test_client.rs +285 -285
- data/vendor/spikard-http/src/testing.rs +377 -377
- data/vendor/spikard-http/src/type_hints.rs +1 -1
- data/vendor/spikard-http/src/validation.rs +1 -1
- data/vendor/spikard-http/src/websocket.rs +324 -324
- data/vendor/spikard-rb/Cargo.toml +42 -42
- data/vendor/spikard-rb/build.rs +8 -8
- data/vendor/spikard-rb/src/background.rs +63 -63
- data/vendor/spikard-rb/src/config.rs +294 -294
- data/vendor/spikard-rb/src/conversion.rs +392 -392
- data/vendor/spikard-rb/src/di.rs +409 -409
- data/vendor/spikard-rb/src/handler.rs +534 -534
- data/vendor/spikard-rb/src/lib.rs +2020 -2020
- data/vendor/spikard-rb/src/lifecycle.rs +267 -267
- data/vendor/spikard-rb/src/server.rs +283 -283
- data/vendor/spikard-rb/src/sse.rs +231 -231
- data/vendor/spikard-rb/src/test_client.rs +404 -404
- data/vendor/spikard-rb/src/test_sse.rs +143 -143
- data/vendor/spikard-rb/src/test_websocket.rs +221 -221
- data/vendor/spikard-rb/src/websocket.rs +233 -233
- metadata +1 -1
|
@@ -1,283 +1,283 @@
|
|
|
1
|
-
//! Value dependency implementation
|
|
2
|
-
//!
|
|
3
|
-
//! This module provides `ValueDependency<T>`, a simple dependency that wraps
|
|
4
|
-
//! a static value and returns it whenever resolved.
|
|
5
|
-
|
|
6
|
-
use super::dependency::Dependency;
|
|
7
|
-
use super::error::DependencyError;
|
|
8
|
-
use super::resolved::ResolvedDependencies;
|
|
9
|
-
use crate::request_data::RequestData;
|
|
10
|
-
use http::Request;
|
|
11
|
-
use std::any::Any;
|
|
12
|
-
use std::future::Future;
|
|
13
|
-
use std::marker::PhantomData;
|
|
14
|
-
use std::pin::Pin;
|
|
15
|
-
use std::sync::Arc;
|
|
16
|
-
|
|
17
|
-
/// A dependency that wraps a static value
|
|
18
|
-
///
|
|
19
|
-
/// This is the simplest form of dependency - it just returns a pre-configured
|
|
20
|
-
/// value whenever resolved. Useful for configuration values, constants, or
|
|
21
|
-
/// pre-built objects.
|
|
22
|
-
///
|
|
23
|
-
/// # Type Parameters
|
|
24
|
-
///
|
|
25
|
-
/// * `T` - The type of value to provide. Must be `Clone + Send + Sync + 'static`.
|
|
26
|
-
///
|
|
27
|
-
/// # Examples
|
|
28
|
-
///
|
|
29
|
-
/// ```ignore
|
|
30
|
-
/// use spikard_core::di::{Dependency, ValueDependency};
|
|
31
|
-
/// use http::Request;
|
|
32
|
-
/// use crate::request_data::RequestData;
|
|
33
|
-
/// use std::collections::HashMap;
|
|
34
|
-
/// use std::sync::Arc;
|
|
35
|
-
///
|
|
36
|
-
/// # tokio_test::block_on(async {
|
|
37
|
-
/// // Create a value dependency with a configuration string
|
|
38
|
-
/// let config = ValueDependency::new("database_url", "postgresql://localhost/mydb");
|
|
39
|
-
///
|
|
40
|
-
/// // Resolve it (returns the same value every time)
|
|
41
|
-
/// let request = Request::builder().body(()).unwrap();
|
|
42
|
-
/// let request_data = RequestData {
|
|
43
|
-
/// path_params: Arc::new(HashMap::new()),
|
|
44
|
-
/// query_params: serde_json::Value::Null,
|
|
45
|
-
/// raw_query_params: Arc::new(HashMap::new()),
|
|
46
|
-
/// body: serde_json::Value::Null,
|
|
47
|
-
/// raw_body: None,
|
|
48
|
-
/// headers: Arc::new(HashMap::new()),
|
|
49
|
-
/// cookies: Arc::new(HashMap::new()),
|
|
50
|
-
/// method: "GET".to_string(),
|
|
51
|
-
/// path: "/".to_string(),
|
|
52
|
-
/// };
|
|
53
|
-
/// let resolved = spikard_core::di::ResolvedDependencies::new();
|
|
54
|
-
///
|
|
55
|
-
/// let result = config.resolve(&request, &request_data, &resolved).await.unwrap();
|
|
56
|
-
/// let value: Arc<String> = result.downcast().unwrap();
|
|
57
|
-
/// assert_eq!(*value, "postgresql://localhost/mydb");
|
|
58
|
-
/// # });
|
|
59
|
-
/// ```
|
|
60
|
-
pub struct ValueDependency<T: Clone + Send + Sync + 'static> {
|
|
61
|
-
key: String,
|
|
62
|
-
value: Arc<T>,
|
|
63
|
-
_phantom: PhantomData<T>,
|
|
64
|
-
}
|
|
65
|
-
|
|
66
|
-
impl<T: Clone + Send + Sync + 'static> ValueDependency<T> {
|
|
67
|
-
/// Create a new value dependency
|
|
68
|
-
///
|
|
69
|
-
/// # Arguments
|
|
70
|
-
///
|
|
71
|
-
/// * `key` - The unique key for this dependency
|
|
72
|
-
/// * `value` - The value to provide when resolved
|
|
73
|
-
///
|
|
74
|
-
/// # Examples
|
|
75
|
-
///
|
|
76
|
-
/// ```ignore
|
|
77
|
-
/// use spikard_core::di::ValueDependency;
|
|
78
|
-
///
|
|
79
|
-
/// // Simple value
|
|
80
|
-
/// let port = ValueDependency::new("port", 8080u16);
|
|
81
|
-
///
|
|
82
|
-
/// // Complex value
|
|
83
|
-
/// #[derive(Clone)]
|
|
84
|
-
/// struct Config {
|
|
85
|
-
/// debug: bool,
|
|
86
|
-
/// timeout: u64,
|
|
87
|
-
/// }
|
|
88
|
-
///
|
|
89
|
-
/// let config = ValueDependency::new("config", Config {
|
|
90
|
-
/// debug: true,
|
|
91
|
-
/// timeout: 30,
|
|
92
|
-
/// });
|
|
93
|
-
/// ```
|
|
94
|
-
pub fn new(key: impl Into<String>, value: T) -> Self {
|
|
95
|
-
Self {
|
|
96
|
-
key: key.into(),
|
|
97
|
-
value: Arc::new(value),
|
|
98
|
-
_phantom: PhantomData,
|
|
99
|
-
}
|
|
100
|
-
}
|
|
101
|
-
}
|
|
102
|
-
|
|
103
|
-
impl<T: Clone + Send + Sync + 'static> Dependency for ValueDependency<T> {
|
|
104
|
-
fn resolve(
|
|
105
|
-
&self,
|
|
106
|
-
_request: &Request<()>,
|
|
107
|
-
_request_data: &RequestData,
|
|
108
|
-
_resolved: &ResolvedDependencies,
|
|
109
|
-
) -> Pin<Box<dyn Future<Output = Result<Arc<dyn Any + Send + Sync>, DependencyError>> + Send>> {
|
|
110
|
-
let value = Arc::clone(&self.value);
|
|
111
|
-
Box::pin(async move { Ok(value as Arc<dyn Any + Send + Sync>) })
|
|
112
|
-
}
|
|
113
|
-
|
|
114
|
-
fn key(&self) -> &str {
|
|
115
|
-
&self.key
|
|
116
|
-
}
|
|
117
|
-
|
|
118
|
-
fn depends_on(&self) -> Vec<String> {
|
|
119
|
-
// Value dependencies have no dependencies
|
|
120
|
-
vec![]
|
|
121
|
-
}
|
|
122
|
-
|
|
123
|
-
fn cacheable(&self) -> bool {
|
|
124
|
-
// Values are inherently cacheable (they never change)
|
|
125
|
-
true
|
|
126
|
-
}
|
|
127
|
-
|
|
128
|
-
fn singleton(&self) -> bool {
|
|
129
|
-
// Values can be singletons (they never change)
|
|
130
|
-
true
|
|
131
|
-
}
|
|
132
|
-
}
|
|
133
|
-
|
|
134
|
-
impl<T: Clone + Send + Sync + 'static> std::fmt::Debug for ValueDependency<T> {
|
|
135
|
-
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
|
136
|
-
f.debug_struct("ValueDependency")
|
|
137
|
-
.field("key", &self.key)
|
|
138
|
-
.field("value_type", &std::any::type_name::<T>())
|
|
139
|
-
.finish()
|
|
140
|
-
}
|
|
141
|
-
}
|
|
142
|
-
|
|
143
|
-
#[cfg(test)]
|
|
144
|
-
mod tests {
|
|
145
|
-
use super::*;
|
|
146
|
-
use std::collections::HashMap;
|
|
147
|
-
|
|
148
|
-
fn make_request_data() -> RequestData {
|
|
149
|
-
RequestData {
|
|
150
|
-
path_params: Arc::new(HashMap::new()),
|
|
151
|
-
query_params: serde_json::Value::Null,
|
|
152
|
-
raw_query_params: Arc::new(HashMap::new()),
|
|
153
|
-
body: serde_json::Value::Null,
|
|
154
|
-
raw_body: None,
|
|
155
|
-
headers: Arc::new(HashMap::new()),
|
|
156
|
-
cookies: Arc::new(HashMap::new()),
|
|
157
|
-
method: "GET".to_string(),
|
|
158
|
-
path: "/".to_string(),
|
|
159
|
-
#[cfg(feature = "di")]
|
|
160
|
-
dependencies: None,
|
|
161
|
-
}
|
|
162
|
-
}
|
|
163
|
-
|
|
164
|
-
#[test]
|
|
165
|
-
fn test_new() {
|
|
166
|
-
let dep = ValueDependency::new("test", 42i32);
|
|
167
|
-
assert_eq!(dep.key(), "test");
|
|
168
|
-
}
|
|
169
|
-
|
|
170
|
-
#[test]
|
|
171
|
-
fn test_key() {
|
|
172
|
-
let dep = ValueDependency::new("my_key", "value");
|
|
173
|
-
assert_eq!(dep.key(), "my_key");
|
|
174
|
-
}
|
|
175
|
-
|
|
176
|
-
#[test]
|
|
177
|
-
fn test_depends_on() {
|
|
178
|
-
let dep = ValueDependency::new("test", 42i32);
|
|
179
|
-
assert_eq!(dep.depends_on(), Vec::<String>::new());
|
|
180
|
-
}
|
|
181
|
-
|
|
182
|
-
#[test]
|
|
183
|
-
fn test_cacheable() {
|
|
184
|
-
let dep = ValueDependency::new("test", 42i32);
|
|
185
|
-
assert!(dep.cacheable());
|
|
186
|
-
}
|
|
187
|
-
|
|
188
|
-
#[test]
|
|
189
|
-
fn test_singleton() {
|
|
190
|
-
let dep = ValueDependency::new("test", 42i32);
|
|
191
|
-
assert!(dep.singleton());
|
|
192
|
-
}
|
|
193
|
-
|
|
194
|
-
#[tokio::test]
|
|
195
|
-
async fn test_resolve_simple() {
|
|
196
|
-
let dep = ValueDependency::new("answer", 42i32);
|
|
197
|
-
let request = Request::builder().body(()).unwrap();
|
|
198
|
-
let request_data = make_request_data();
|
|
199
|
-
let resolved = ResolvedDependencies::new();
|
|
200
|
-
|
|
201
|
-
let result = dep.resolve(&request, &request_data, &resolved).await;
|
|
202
|
-
assert!(result.is_ok());
|
|
203
|
-
|
|
204
|
-
let value: Arc<i32> = result.unwrap().downcast().unwrap();
|
|
205
|
-
assert_eq!(*value, 42);
|
|
206
|
-
}
|
|
207
|
-
|
|
208
|
-
#[tokio::test]
|
|
209
|
-
async fn test_resolve_string() {
|
|
210
|
-
let dep = ValueDependency::new("message", "Hello, World!".to_string());
|
|
211
|
-
let request = Request::builder().body(()).unwrap();
|
|
212
|
-
let request_data = make_request_data();
|
|
213
|
-
let resolved = ResolvedDependencies::new();
|
|
214
|
-
|
|
215
|
-
let result = dep.resolve(&request, &request_data, &resolved).await;
|
|
216
|
-
assert!(result.is_ok());
|
|
217
|
-
|
|
218
|
-
let value: Arc<String> = result.unwrap().downcast().unwrap();
|
|
219
|
-
assert_eq!(*value, "Hello, World!");
|
|
220
|
-
}
|
|
221
|
-
|
|
222
|
-
#[tokio::test]
|
|
223
|
-
async fn test_resolve_concurrent() {
|
|
224
|
-
let dep = Arc::new(ValueDependency::new("shared", 100i32));
|
|
225
|
-
let request = Request::builder().body(()).unwrap();
|
|
226
|
-
let request_data = make_request_data();
|
|
227
|
-
|
|
228
|
-
// Resolve concurrently from multiple tasks
|
|
229
|
-
let handles: Vec<_> = (0..10)
|
|
230
|
-
.map(|_| {
|
|
231
|
-
let dep = Arc::clone(&dep);
|
|
232
|
-
let req = request.clone();
|
|
233
|
-
let data = request_data.clone();
|
|
234
|
-
tokio::spawn(async move {
|
|
235
|
-
let resolved = ResolvedDependencies::new();
|
|
236
|
-
let result = dep.resolve(&req, &data, &resolved).await.unwrap();
|
|
237
|
-
let value: Arc<i32> = result.downcast().unwrap();
|
|
238
|
-
*value
|
|
239
|
-
})
|
|
240
|
-
})
|
|
241
|
-
.collect();
|
|
242
|
-
|
|
243
|
-
for handle in handles {
|
|
244
|
-
let value = handle.await.unwrap();
|
|
245
|
-
assert_eq!(value, 100);
|
|
246
|
-
}
|
|
247
|
-
}
|
|
248
|
-
|
|
249
|
-
#[derive(Clone, Debug, PartialEq)]
|
|
250
|
-
struct ComplexValue {
|
|
251
|
-
name: String,
|
|
252
|
-
count: i32,
|
|
253
|
-
tags: Vec<String>,
|
|
254
|
-
}
|
|
255
|
-
|
|
256
|
-
#[tokio::test]
|
|
257
|
-
async fn test_resolve_complex_type() {
|
|
258
|
-
let complex = ComplexValue {
|
|
259
|
-
name: "test".to_string(),
|
|
260
|
-
count: 42,
|
|
261
|
-
tags: vec!["tag1".to_string(), "tag2".to_string()],
|
|
262
|
-
};
|
|
263
|
-
|
|
264
|
-
let dep = ValueDependency::new("complex", complex.clone());
|
|
265
|
-
let request = Request::builder().body(()).unwrap();
|
|
266
|
-
let request_data = make_request_data();
|
|
267
|
-
let resolved = ResolvedDependencies::new();
|
|
268
|
-
|
|
269
|
-
let result = dep.resolve(&request, &request_data, &resolved).await;
|
|
270
|
-
assert!(result.is_ok());
|
|
271
|
-
|
|
272
|
-
let value: Arc<ComplexValue> = result.unwrap().downcast().unwrap();
|
|
273
|
-
assert_eq!(*value, complex);
|
|
274
|
-
}
|
|
275
|
-
|
|
276
|
-
#[test]
|
|
277
|
-
fn test_debug() {
|
|
278
|
-
let dep = ValueDependency::new("test", 42i32);
|
|
279
|
-
let debug_str = format!("{:?}", dep);
|
|
280
|
-
assert!(debug_str.contains("ValueDependency"));
|
|
281
|
-
assert!(debug_str.contains("test"));
|
|
282
|
-
}
|
|
283
|
-
}
|
|
1
|
+
//! Value dependency implementation
|
|
2
|
+
//!
|
|
3
|
+
//! This module provides `ValueDependency<T>`, a simple dependency that wraps
|
|
4
|
+
//! a static value and returns it whenever resolved.
|
|
5
|
+
|
|
6
|
+
use super::dependency::Dependency;
|
|
7
|
+
use super::error::DependencyError;
|
|
8
|
+
use super::resolved::ResolvedDependencies;
|
|
9
|
+
use crate::request_data::RequestData;
|
|
10
|
+
use http::Request;
|
|
11
|
+
use std::any::Any;
|
|
12
|
+
use std::future::Future;
|
|
13
|
+
use std::marker::PhantomData;
|
|
14
|
+
use std::pin::Pin;
|
|
15
|
+
use std::sync::Arc;
|
|
16
|
+
|
|
17
|
+
/// A dependency that wraps a static value
|
|
18
|
+
///
|
|
19
|
+
/// This is the simplest form of dependency - it just returns a pre-configured
|
|
20
|
+
/// value whenever resolved. Useful for configuration values, constants, or
|
|
21
|
+
/// pre-built objects.
|
|
22
|
+
///
|
|
23
|
+
/// # Type Parameters
|
|
24
|
+
///
|
|
25
|
+
/// * `T` - The type of value to provide. Must be `Clone + Send + Sync + 'static`.
|
|
26
|
+
///
|
|
27
|
+
/// # Examples
|
|
28
|
+
///
|
|
29
|
+
/// ```ignore
|
|
30
|
+
/// use spikard_core::di::{Dependency, ValueDependency};
|
|
31
|
+
/// use http::Request;
|
|
32
|
+
/// use crate::request_data::RequestData;
|
|
33
|
+
/// use std::collections::HashMap;
|
|
34
|
+
/// use std::sync::Arc;
|
|
35
|
+
///
|
|
36
|
+
/// # tokio_test::block_on(async {
|
|
37
|
+
/// // Create a value dependency with a configuration string
|
|
38
|
+
/// let config = ValueDependency::new("database_url", "postgresql://localhost/mydb");
|
|
39
|
+
///
|
|
40
|
+
/// // Resolve it (returns the same value every time)
|
|
41
|
+
/// let request = Request::builder().body(()).unwrap();
|
|
42
|
+
/// let request_data = RequestData {
|
|
43
|
+
/// path_params: Arc::new(HashMap::new()),
|
|
44
|
+
/// query_params: serde_json::Value::Null,
|
|
45
|
+
/// raw_query_params: Arc::new(HashMap::new()),
|
|
46
|
+
/// body: serde_json::Value::Null,
|
|
47
|
+
/// raw_body: None,
|
|
48
|
+
/// headers: Arc::new(HashMap::new()),
|
|
49
|
+
/// cookies: Arc::new(HashMap::new()),
|
|
50
|
+
/// method: "GET".to_string(),
|
|
51
|
+
/// path: "/".to_string(),
|
|
52
|
+
/// };
|
|
53
|
+
/// let resolved = spikard_core::di::ResolvedDependencies::new();
|
|
54
|
+
///
|
|
55
|
+
/// let result = config.resolve(&request, &request_data, &resolved).await.unwrap();
|
|
56
|
+
/// let value: Arc<String> = result.downcast().unwrap();
|
|
57
|
+
/// assert_eq!(*value, "postgresql://localhost/mydb");
|
|
58
|
+
/// # });
|
|
59
|
+
/// ```
|
|
60
|
+
pub struct ValueDependency<T: Clone + Send + Sync + 'static> {
|
|
61
|
+
key: String,
|
|
62
|
+
value: Arc<T>,
|
|
63
|
+
_phantom: PhantomData<T>,
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
impl<T: Clone + Send + Sync + 'static> ValueDependency<T> {
|
|
67
|
+
/// Create a new value dependency
|
|
68
|
+
///
|
|
69
|
+
/// # Arguments
|
|
70
|
+
///
|
|
71
|
+
/// * `key` - The unique key for this dependency
|
|
72
|
+
/// * `value` - The value to provide when resolved
|
|
73
|
+
///
|
|
74
|
+
/// # Examples
|
|
75
|
+
///
|
|
76
|
+
/// ```ignore
|
|
77
|
+
/// use spikard_core::di::ValueDependency;
|
|
78
|
+
///
|
|
79
|
+
/// // Simple value
|
|
80
|
+
/// let port = ValueDependency::new("port", 8080u16);
|
|
81
|
+
///
|
|
82
|
+
/// // Complex value
|
|
83
|
+
/// #[derive(Clone)]
|
|
84
|
+
/// struct Config {
|
|
85
|
+
/// debug: bool,
|
|
86
|
+
/// timeout: u64,
|
|
87
|
+
/// }
|
|
88
|
+
///
|
|
89
|
+
/// let config = ValueDependency::new("config", Config {
|
|
90
|
+
/// debug: true,
|
|
91
|
+
/// timeout: 30,
|
|
92
|
+
/// });
|
|
93
|
+
/// ```
|
|
94
|
+
pub fn new(key: impl Into<String>, value: T) -> Self {
|
|
95
|
+
Self {
|
|
96
|
+
key: key.into(),
|
|
97
|
+
value: Arc::new(value),
|
|
98
|
+
_phantom: PhantomData,
|
|
99
|
+
}
|
|
100
|
+
}
|
|
101
|
+
}
|
|
102
|
+
|
|
103
|
+
impl<T: Clone + Send + Sync + 'static> Dependency for ValueDependency<T> {
|
|
104
|
+
fn resolve(
|
|
105
|
+
&self,
|
|
106
|
+
_request: &Request<()>,
|
|
107
|
+
_request_data: &RequestData,
|
|
108
|
+
_resolved: &ResolvedDependencies,
|
|
109
|
+
) -> Pin<Box<dyn Future<Output = Result<Arc<dyn Any + Send + Sync>, DependencyError>> + Send>> {
|
|
110
|
+
let value = Arc::clone(&self.value);
|
|
111
|
+
Box::pin(async move { Ok(value as Arc<dyn Any + Send + Sync>) })
|
|
112
|
+
}
|
|
113
|
+
|
|
114
|
+
fn key(&self) -> &str {
|
|
115
|
+
&self.key
|
|
116
|
+
}
|
|
117
|
+
|
|
118
|
+
fn depends_on(&self) -> Vec<String> {
|
|
119
|
+
// Value dependencies have no dependencies
|
|
120
|
+
vec![]
|
|
121
|
+
}
|
|
122
|
+
|
|
123
|
+
fn cacheable(&self) -> bool {
|
|
124
|
+
// Values are inherently cacheable (they never change)
|
|
125
|
+
true
|
|
126
|
+
}
|
|
127
|
+
|
|
128
|
+
fn singleton(&self) -> bool {
|
|
129
|
+
// Values can be singletons (they never change)
|
|
130
|
+
true
|
|
131
|
+
}
|
|
132
|
+
}
|
|
133
|
+
|
|
134
|
+
impl<T: Clone + Send + Sync + 'static> std::fmt::Debug for ValueDependency<T> {
|
|
135
|
+
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
|
136
|
+
f.debug_struct("ValueDependency")
|
|
137
|
+
.field("key", &self.key)
|
|
138
|
+
.field("value_type", &std::any::type_name::<T>())
|
|
139
|
+
.finish()
|
|
140
|
+
}
|
|
141
|
+
}
|
|
142
|
+
|
|
143
|
+
#[cfg(test)]
|
|
144
|
+
mod tests {
|
|
145
|
+
use super::*;
|
|
146
|
+
use std::collections::HashMap;
|
|
147
|
+
|
|
148
|
+
fn make_request_data() -> RequestData {
|
|
149
|
+
RequestData {
|
|
150
|
+
path_params: Arc::new(HashMap::new()),
|
|
151
|
+
query_params: serde_json::Value::Null,
|
|
152
|
+
raw_query_params: Arc::new(HashMap::new()),
|
|
153
|
+
body: serde_json::Value::Null,
|
|
154
|
+
raw_body: None,
|
|
155
|
+
headers: Arc::new(HashMap::new()),
|
|
156
|
+
cookies: Arc::new(HashMap::new()),
|
|
157
|
+
method: "GET".to_string(),
|
|
158
|
+
path: "/".to_string(),
|
|
159
|
+
#[cfg(feature = "di")]
|
|
160
|
+
dependencies: None,
|
|
161
|
+
}
|
|
162
|
+
}
|
|
163
|
+
|
|
164
|
+
#[test]
|
|
165
|
+
fn test_new() {
|
|
166
|
+
let dep = ValueDependency::new("test", 42i32);
|
|
167
|
+
assert_eq!(dep.key(), "test");
|
|
168
|
+
}
|
|
169
|
+
|
|
170
|
+
#[test]
|
|
171
|
+
fn test_key() {
|
|
172
|
+
let dep = ValueDependency::new("my_key", "value");
|
|
173
|
+
assert_eq!(dep.key(), "my_key");
|
|
174
|
+
}
|
|
175
|
+
|
|
176
|
+
#[test]
|
|
177
|
+
fn test_depends_on() {
|
|
178
|
+
let dep = ValueDependency::new("test", 42i32);
|
|
179
|
+
assert_eq!(dep.depends_on(), Vec::<String>::new());
|
|
180
|
+
}
|
|
181
|
+
|
|
182
|
+
#[test]
|
|
183
|
+
fn test_cacheable() {
|
|
184
|
+
let dep = ValueDependency::new("test", 42i32);
|
|
185
|
+
assert!(dep.cacheable());
|
|
186
|
+
}
|
|
187
|
+
|
|
188
|
+
#[test]
|
|
189
|
+
fn test_singleton() {
|
|
190
|
+
let dep = ValueDependency::new("test", 42i32);
|
|
191
|
+
assert!(dep.singleton());
|
|
192
|
+
}
|
|
193
|
+
|
|
194
|
+
#[tokio::test]
|
|
195
|
+
async fn test_resolve_simple() {
|
|
196
|
+
let dep = ValueDependency::new("answer", 42i32);
|
|
197
|
+
let request = Request::builder().body(()).unwrap();
|
|
198
|
+
let request_data = make_request_data();
|
|
199
|
+
let resolved = ResolvedDependencies::new();
|
|
200
|
+
|
|
201
|
+
let result = dep.resolve(&request, &request_data, &resolved).await;
|
|
202
|
+
assert!(result.is_ok());
|
|
203
|
+
|
|
204
|
+
let value: Arc<i32> = result.unwrap().downcast().unwrap();
|
|
205
|
+
assert_eq!(*value, 42);
|
|
206
|
+
}
|
|
207
|
+
|
|
208
|
+
#[tokio::test]
|
|
209
|
+
async fn test_resolve_string() {
|
|
210
|
+
let dep = ValueDependency::new("message", "Hello, World!".to_string());
|
|
211
|
+
let request = Request::builder().body(()).unwrap();
|
|
212
|
+
let request_data = make_request_data();
|
|
213
|
+
let resolved = ResolvedDependencies::new();
|
|
214
|
+
|
|
215
|
+
let result = dep.resolve(&request, &request_data, &resolved).await;
|
|
216
|
+
assert!(result.is_ok());
|
|
217
|
+
|
|
218
|
+
let value: Arc<String> = result.unwrap().downcast().unwrap();
|
|
219
|
+
assert_eq!(*value, "Hello, World!");
|
|
220
|
+
}
|
|
221
|
+
|
|
222
|
+
#[tokio::test]
|
|
223
|
+
async fn test_resolve_concurrent() {
|
|
224
|
+
let dep = Arc::new(ValueDependency::new("shared", 100i32));
|
|
225
|
+
let request = Request::builder().body(()).unwrap();
|
|
226
|
+
let request_data = make_request_data();
|
|
227
|
+
|
|
228
|
+
// Resolve concurrently from multiple tasks
|
|
229
|
+
let handles: Vec<_> = (0..10)
|
|
230
|
+
.map(|_| {
|
|
231
|
+
let dep = Arc::clone(&dep);
|
|
232
|
+
let req = request.clone();
|
|
233
|
+
let data = request_data.clone();
|
|
234
|
+
tokio::spawn(async move {
|
|
235
|
+
let resolved = ResolvedDependencies::new();
|
|
236
|
+
let result = dep.resolve(&req, &data, &resolved).await.unwrap();
|
|
237
|
+
let value: Arc<i32> = result.downcast().unwrap();
|
|
238
|
+
*value
|
|
239
|
+
})
|
|
240
|
+
})
|
|
241
|
+
.collect();
|
|
242
|
+
|
|
243
|
+
for handle in handles {
|
|
244
|
+
let value = handle.await.unwrap();
|
|
245
|
+
assert_eq!(value, 100);
|
|
246
|
+
}
|
|
247
|
+
}
|
|
248
|
+
|
|
249
|
+
#[derive(Clone, Debug, PartialEq)]
|
|
250
|
+
struct ComplexValue {
|
|
251
|
+
name: String,
|
|
252
|
+
count: i32,
|
|
253
|
+
tags: Vec<String>,
|
|
254
|
+
}
|
|
255
|
+
|
|
256
|
+
#[tokio::test]
|
|
257
|
+
async fn test_resolve_complex_type() {
|
|
258
|
+
let complex = ComplexValue {
|
|
259
|
+
name: "test".to_string(),
|
|
260
|
+
count: 42,
|
|
261
|
+
tags: vec!["tag1".to_string(), "tag2".to_string()],
|
|
262
|
+
};
|
|
263
|
+
|
|
264
|
+
let dep = ValueDependency::new("complex", complex.clone());
|
|
265
|
+
let request = Request::builder().body(()).unwrap();
|
|
266
|
+
let request_data = make_request_data();
|
|
267
|
+
let resolved = ResolvedDependencies::new();
|
|
268
|
+
|
|
269
|
+
let result = dep.resolve(&request, &request_data, &resolved).await;
|
|
270
|
+
assert!(result.is_ok());
|
|
271
|
+
|
|
272
|
+
let value: Arc<ComplexValue> = result.unwrap().downcast().unwrap();
|
|
273
|
+
assert_eq!(*value, complex);
|
|
274
|
+
}
|
|
275
|
+
|
|
276
|
+
#[test]
|
|
277
|
+
fn test_debug() {
|
|
278
|
+
let dep = ValueDependency::new("test", 42i32);
|
|
279
|
+
let debug_str = format!("{:?}", dep);
|
|
280
|
+
assert!(debug_str.contains("ValueDependency"));
|
|
281
|
+
assert!(debug_str.contains("test"));
|
|
282
|
+
}
|
|
283
|
+
}
|
|
@@ -1,39 +1,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
|
-
}
|
|
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
|
+
}
|