spikard 0.4.0-x86_64-linux
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 +7 -0
- data/LICENSE +1 -0
- data/README.md +659 -0
- data/ext/spikard_rb/Cargo.toml +17 -0
- data/ext/spikard_rb/extconf.rb +10 -0
- data/ext/spikard_rb/src/lib.rs +6 -0
- data/lib/spikard/app.rb +405 -0
- data/lib/spikard/background.rb +27 -0
- data/lib/spikard/config.rb +396 -0
- data/lib/spikard/converters.rb +13 -0
- data/lib/spikard/handler_wrapper.rb +113 -0
- data/lib/spikard/provide.rb +214 -0
- data/lib/spikard/response.rb +173 -0
- data/lib/spikard/schema.rb +243 -0
- data/lib/spikard/sse.rb +111 -0
- data/lib/spikard/streaming_response.rb +44 -0
- data/lib/spikard/testing.rb +221 -0
- data/lib/spikard/upload_file.rb +131 -0
- data/lib/spikard/version.rb +5 -0
- data/lib/spikard/websocket.rb +59 -0
- data/lib/spikard.rb +43 -0
- data/sig/spikard.rbs +366 -0
- data/vendor/bundle/ruby/3.4.0/gems/diff-lcs-1.6.2/mise.toml +5 -0
- data/vendor/bundle/ruby/3.4.0/gems/rake-compiler-dock-1.10.0/build/buildkitd.toml +2 -0
- data/vendor/crates/spikard-bindings-shared/Cargo.toml +63 -0
- data/vendor/crates/spikard-bindings-shared/examples/config_extraction.rs +139 -0
- data/vendor/crates/spikard-bindings-shared/src/config_extractor.rs +561 -0
- data/vendor/crates/spikard-bindings-shared/src/conversion_traits.rs +194 -0
- data/vendor/crates/spikard-bindings-shared/src/di_traits.rs +246 -0
- data/vendor/crates/spikard-bindings-shared/src/error_response.rs +403 -0
- data/vendor/crates/spikard-bindings-shared/src/handler_base.rs +274 -0
- data/vendor/crates/spikard-bindings-shared/src/lib.rs +25 -0
- data/vendor/crates/spikard-bindings-shared/src/lifecycle_base.rs +298 -0
- data/vendor/crates/spikard-bindings-shared/src/lifecycle_executor.rs +637 -0
- data/vendor/crates/spikard-bindings-shared/src/response_builder.rs +309 -0
- data/vendor/crates/spikard-bindings-shared/src/test_client_base.rs +248 -0
- data/vendor/crates/spikard-bindings-shared/src/validation_helpers.rs +355 -0
- data/vendor/crates/spikard-bindings-shared/tests/comprehensive_coverage.rs +502 -0
- data/vendor/crates/spikard-bindings-shared/tests/error_response_edge_cases.rs +389 -0
- data/vendor/crates/spikard-bindings-shared/tests/handler_base_integration.rs +413 -0
- data/vendor/crates/spikard-core/Cargo.toml +40 -0
- data/vendor/crates/spikard-core/src/bindings/mod.rs +3 -0
- data/vendor/crates/spikard-core/src/bindings/response.rs +133 -0
- data/vendor/crates/spikard-core/src/debug.rs +63 -0
- data/vendor/crates/spikard-core/src/di/container.rs +726 -0
- data/vendor/crates/spikard-core/src/di/dependency.rs +273 -0
- data/vendor/crates/spikard-core/src/di/error.rs +118 -0
- data/vendor/crates/spikard-core/src/di/factory.rs +538 -0
- data/vendor/crates/spikard-core/src/di/graph.rs +545 -0
- data/vendor/crates/spikard-core/src/di/mod.rs +192 -0
- data/vendor/crates/spikard-core/src/di/resolved.rs +411 -0
- data/vendor/crates/spikard-core/src/di/value.rs +283 -0
- data/vendor/crates/spikard-core/src/errors.rs +39 -0
- data/vendor/crates/spikard-core/src/http.rs +153 -0
- data/vendor/crates/spikard-core/src/lib.rs +29 -0
- data/vendor/crates/spikard-core/src/lifecycle.rs +422 -0
- data/vendor/crates/spikard-core/src/metadata.rs +397 -0
- data/vendor/crates/spikard-core/src/parameters.rs +723 -0
- data/vendor/crates/spikard-core/src/problem.rs +310 -0
- data/vendor/crates/spikard-core/src/request_data.rs +189 -0
- data/vendor/crates/spikard-core/src/router.rs +249 -0
- data/vendor/crates/spikard-core/src/schema_registry.rs +183 -0
- data/vendor/crates/spikard-core/src/type_hints.rs +304 -0
- data/vendor/crates/spikard-core/src/validation/error_mapper.rs +689 -0
- data/vendor/crates/spikard-core/src/validation/mod.rs +459 -0
- data/vendor/crates/spikard-http/Cargo.toml +58 -0
- data/vendor/crates/spikard-http/examples/sse-notifications.rs +147 -0
- data/vendor/crates/spikard-http/examples/websocket-chat.rs +91 -0
- data/vendor/crates/spikard-http/src/auth.rs +247 -0
- data/vendor/crates/spikard-http/src/background.rs +1562 -0
- data/vendor/crates/spikard-http/src/bindings/mod.rs +3 -0
- data/vendor/crates/spikard-http/src/bindings/response.rs +1 -0
- data/vendor/crates/spikard-http/src/body_metadata.rs +8 -0
- data/vendor/crates/spikard-http/src/cors.rs +490 -0
- data/vendor/crates/spikard-http/src/debug.rs +63 -0
- data/vendor/crates/spikard-http/src/di_handler.rs +1878 -0
- data/vendor/crates/spikard-http/src/handler_response.rs +532 -0
- data/vendor/crates/spikard-http/src/handler_trait.rs +861 -0
- data/vendor/crates/spikard-http/src/handler_trait_tests.rs +284 -0
- data/vendor/crates/spikard-http/src/lib.rs +524 -0
- data/vendor/crates/spikard-http/src/lifecycle/adapter.rs +149 -0
- data/vendor/crates/spikard-http/src/lifecycle.rs +428 -0
- data/vendor/crates/spikard-http/src/middleware/mod.rs +285 -0
- data/vendor/crates/spikard-http/src/middleware/multipart.rs +930 -0
- data/vendor/crates/spikard-http/src/middleware/urlencoded.rs +541 -0
- data/vendor/crates/spikard-http/src/middleware/validation.rs +287 -0
- data/vendor/crates/spikard-http/src/openapi/mod.rs +309 -0
- data/vendor/crates/spikard-http/src/openapi/parameter_extraction.rs +535 -0
- data/vendor/crates/spikard-http/src/openapi/schema_conversion.rs +867 -0
- data/vendor/crates/spikard-http/src/openapi/spec_generation.rs +678 -0
- data/vendor/crates/spikard-http/src/query_parser.rs +369 -0
- data/vendor/crates/spikard-http/src/response.rs +399 -0
- data/vendor/crates/spikard-http/src/server/handler.rs +1557 -0
- data/vendor/crates/spikard-http/src/server/lifecycle_execution.rs +98 -0
- data/vendor/crates/spikard-http/src/server/mod.rs +806 -0
- data/vendor/crates/spikard-http/src/server/request_extraction.rs +630 -0
- data/vendor/crates/spikard-http/src/server/routing_factory.rs +497 -0
- data/vendor/crates/spikard-http/src/sse.rs +961 -0
- data/vendor/crates/spikard-http/src/testing/form.rs +14 -0
- data/vendor/crates/spikard-http/src/testing/multipart.rs +60 -0
- data/vendor/crates/spikard-http/src/testing/test_client.rs +285 -0
- data/vendor/crates/spikard-http/src/testing.rs +377 -0
- data/vendor/crates/spikard-http/src/websocket.rs +831 -0
- data/vendor/crates/spikard-http/tests/background_behavior.rs +918 -0
- data/vendor/crates/spikard-http/tests/common/handlers.rs +308 -0
- data/vendor/crates/spikard-http/tests/common/mod.rs +21 -0
- data/vendor/crates/spikard-http/tests/di_integration.rs +202 -0
- data/vendor/crates/spikard-http/tests/doc_snippets.rs +4 -0
- data/vendor/crates/spikard-http/tests/lifecycle_execution.rs +1135 -0
- data/vendor/crates/spikard-http/tests/multipart_behavior.rs +688 -0
- data/vendor/crates/spikard-http/tests/server_config_builder.rs +324 -0
- data/vendor/crates/spikard-http/tests/sse_behavior.rs +728 -0
- data/vendor/crates/spikard-http/tests/websocket_behavior.rs +724 -0
- data/vendor/crates/spikard-rb/Cargo.toml +43 -0
- data/vendor/crates/spikard-rb/build.rs +199 -0
- data/vendor/crates/spikard-rb/src/background.rs +63 -0
- data/vendor/crates/spikard-rb/src/config/mod.rs +5 -0
- data/vendor/crates/spikard-rb/src/config/server_config.rs +283 -0
- data/vendor/crates/spikard-rb/src/conversion.rs +459 -0
- data/vendor/crates/spikard-rb/src/di/builder.rs +105 -0
- data/vendor/crates/spikard-rb/src/di/mod.rs +413 -0
- data/vendor/crates/spikard-rb/src/handler.rs +612 -0
- data/vendor/crates/spikard-rb/src/integration/mod.rs +3 -0
- data/vendor/crates/spikard-rb/src/lib.rs +1857 -0
- data/vendor/crates/spikard-rb/src/lifecycle.rs +275 -0
- data/vendor/crates/spikard-rb/src/metadata/mod.rs +5 -0
- data/vendor/crates/spikard-rb/src/metadata/route_extraction.rs +427 -0
- data/vendor/crates/spikard-rb/src/runtime/mod.rs +5 -0
- data/vendor/crates/spikard-rb/src/runtime/server_runner.rs +326 -0
- data/vendor/crates/spikard-rb/src/server.rs +283 -0
- data/vendor/crates/spikard-rb/src/sse.rs +231 -0
- data/vendor/crates/spikard-rb/src/testing/client.rs +404 -0
- data/vendor/crates/spikard-rb/src/testing/mod.rs +7 -0
- data/vendor/crates/spikard-rb/src/testing/sse.rs +143 -0
- data/vendor/crates/spikard-rb/src/testing/websocket.rs +221 -0
- data/vendor/crates/spikard-rb/src/websocket.rs +233 -0
- data/vendor/crates/spikard-rb/tests/magnus_ffi_tests.rs +14 -0
- metadata +213 -0
|
@@ -0,0 +1,724 @@
|
|
|
1
|
+
//! Behavioral tests for WebSocket functionality in spikard-http
|
|
2
|
+
//!
|
|
3
|
+
//! This test module verifies observable WebSocket behavior from the perspective of
|
|
4
|
+
//! external clients and handlers, focusing on:
|
|
5
|
+
//!
|
|
6
|
+
//! 1. **Connection Lifecycle** - Proper initialization, connection establishment,
|
|
7
|
+
//! and graceful disconnection with lifecycle hook invocation
|
|
8
|
+
//!
|
|
9
|
+
//! 2. **Concurrent Message Handling** - Multiple messages in rapid succession and
|
|
10
|
+
//! from concurrent tasks, ensuring thread-safety and isolation between connections
|
|
11
|
+
//!
|
|
12
|
+
//! 3. **Message Ordering** - Sequential delivery guarantees and ordering preservation
|
|
13
|
+
//! under various load conditions
|
|
14
|
+
//!
|
|
15
|
+
//! 4. **Abort Handling** - Recovery from errors, state preservation across errors,
|
|
16
|
+
//! and proper error propagation
|
|
17
|
+
//!
|
|
18
|
+
//! 5. **Schema Validation** - JSON schema validation of incoming messages with
|
|
19
|
+
//! rejection of malformed payloads
|
|
20
|
+
//!
|
|
21
|
+
//! 6. **Handler Error Propagation** - Proper handling of errors returned by handlers,
|
|
22
|
+
//! error state transitions, and recovery mechanisms
|
|
23
|
+
//!
|
|
24
|
+
//! 7. **Message Buffering** - Behavior under high load, rapid message bursts,
|
|
25
|
+
//! and concurrent client processing
|
|
26
|
+
//!
|
|
27
|
+
//! These tests verify behavioral contracts without testing implementation details,
|
|
28
|
+
//! ensuring the WebSocket module works correctly when used as a black box.
|
|
29
|
+
|
|
30
|
+
use serde_json::{Value, json};
|
|
31
|
+
use spikard_http::websocket::{WebSocketHandler, WebSocketState};
|
|
32
|
+
use std::sync::atomic::{AtomicBool, AtomicUsize, Ordering};
|
|
33
|
+
use std::sync::{Arc, Mutex};
|
|
34
|
+
use std::time::Duration;
|
|
35
|
+
use tokio::time::sleep;
|
|
36
|
+
|
|
37
|
+
// ===== Test Handlers =====
|
|
38
|
+
|
|
39
|
+
/// Handler that echoes messages back to the client
|
|
40
|
+
#[derive(Debug, Clone)]
|
|
41
|
+
struct EchoHandler;
|
|
42
|
+
|
|
43
|
+
impl WebSocketHandler for EchoHandler {
|
|
44
|
+
async fn handle_message(&self, message: Value) -> Option<Value> {
|
|
45
|
+
Some(message)
|
|
46
|
+
}
|
|
47
|
+
}
|
|
48
|
+
|
|
49
|
+
/// Handler that tracks lifecycle events and message count
|
|
50
|
+
#[derive(Debug, Clone)]
|
|
51
|
+
struct LifecycleHandler {
|
|
52
|
+
connect_called: Arc<AtomicBool>,
|
|
53
|
+
disconnect_called: Arc<AtomicBool>,
|
|
54
|
+
message_count: Arc<AtomicUsize>,
|
|
55
|
+
messages: Arc<Mutex<Vec<Value>>>,
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
impl LifecycleHandler {
|
|
59
|
+
fn new() -> Self {
|
|
60
|
+
Self {
|
|
61
|
+
connect_called: Arc::new(AtomicBool::new(false)),
|
|
62
|
+
disconnect_called: Arc::new(AtomicBool::new(false)),
|
|
63
|
+
message_count: Arc::new(AtomicUsize::new(0)),
|
|
64
|
+
messages: Arc::new(Mutex::new(Vec::new())),
|
|
65
|
+
}
|
|
66
|
+
}
|
|
67
|
+
|
|
68
|
+
fn reset(&self) {
|
|
69
|
+
self.connect_called.store(false, Ordering::SeqCst);
|
|
70
|
+
self.disconnect_called.store(false, Ordering::SeqCst);
|
|
71
|
+
self.message_count.store(0, Ordering::SeqCst);
|
|
72
|
+
self.messages.lock().unwrap().clear();
|
|
73
|
+
}
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
impl WebSocketHandler for LifecycleHandler {
|
|
77
|
+
async fn handle_message(&self, message: Value) -> Option<Value> {
|
|
78
|
+
self.message_count.fetch_add(1, Ordering::SeqCst);
|
|
79
|
+
self.messages.lock().unwrap().push(message.clone());
|
|
80
|
+
Some(message)
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
async fn on_connect(&self) {
|
|
84
|
+
self.connect_called.store(true, Ordering::SeqCst);
|
|
85
|
+
}
|
|
86
|
+
|
|
87
|
+
async fn on_disconnect(&self) {
|
|
88
|
+
self.disconnect_called.store(true, Ordering::SeqCst);
|
|
89
|
+
}
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
/// Handler that validates message schema
|
|
93
|
+
#[derive(Debug)]
|
|
94
|
+
struct SchemaValidationHandler {
|
|
95
|
+
invalid_messages: Arc<Mutex<Vec<String>>>,
|
|
96
|
+
}
|
|
97
|
+
|
|
98
|
+
impl SchemaValidationHandler {
|
|
99
|
+
fn new() -> Self {
|
|
100
|
+
Self {
|
|
101
|
+
invalid_messages: Arc::new(Mutex::new(Vec::new())),
|
|
102
|
+
}
|
|
103
|
+
}
|
|
104
|
+
}
|
|
105
|
+
|
|
106
|
+
impl WebSocketHandler for SchemaValidationHandler {
|
|
107
|
+
async fn handle_message(&self, message: Value) -> Option<Value> {
|
|
108
|
+
// Only accept messages with a "type" field
|
|
109
|
+
if message.get("type").is_some() {
|
|
110
|
+
Some(json!({"status": "ok", "echo": message}))
|
|
111
|
+
} else {
|
|
112
|
+
self.invalid_messages.lock().unwrap().push(format!("{:?}", message));
|
|
113
|
+
None
|
|
114
|
+
}
|
|
115
|
+
}
|
|
116
|
+
}
|
|
117
|
+
|
|
118
|
+
/// Handler that simulates errors during message processing
|
|
119
|
+
#[derive(Debug)]
|
|
120
|
+
struct ErrorHandler {
|
|
121
|
+
error_count: Arc<AtomicUsize>,
|
|
122
|
+
should_error: Arc<AtomicBool>,
|
|
123
|
+
}
|
|
124
|
+
|
|
125
|
+
impl ErrorHandler {
|
|
126
|
+
fn new() -> Self {
|
|
127
|
+
Self {
|
|
128
|
+
error_count: Arc::new(AtomicUsize::new(0)),
|
|
129
|
+
should_error: Arc::new(AtomicBool::new(false)),
|
|
130
|
+
}
|
|
131
|
+
}
|
|
132
|
+
}
|
|
133
|
+
|
|
134
|
+
impl WebSocketHandler for ErrorHandler {
|
|
135
|
+
async fn handle_message(&self, message: Value) -> Option<Value> {
|
|
136
|
+
if self.should_error.load(Ordering::SeqCst) {
|
|
137
|
+
self.error_count.fetch_add(1, Ordering::SeqCst);
|
|
138
|
+
// Return None to simulate error (no response)
|
|
139
|
+
None
|
|
140
|
+
} else {
|
|
141
|
+
Some(message)
|
|
142
|
+
}
|
|
143
|
+
}
|
|
144
|
+
}
|
|
145
|
+
|
|
146
|
+
/// Handler that tracks message ordering
|
|
147
|
+
#[derive(Debug)]
|
|
148
|
+
struct OrderingHandler {
|
|
149
|
+
messages: Arc<Mutex<Vec<usize>>>,
|
|
150
|
+
}
|
|
151
|
+
|
|
152
|
+
impl OrderingHandler {
|
|
153
|
+
fn new() -> Self {
|
|
154
|
+
Self {
|
|
155
|
+
messages: Arc::new(Mutex::new(Vec::new())),
|
|
156
|
+
}
|
|
157
|
+
}
|
|
158
|
+
}
|
|
159
|
+
|
|
160
|
+
impl WebSocketHandler for OrderingHandler {
|
|
161
|
+
async fn handle_message(&self, message: Value) -> Option<Value> {
|
|
162
|
+
if let Some(seq) = message.get("sequence").and_then(|v| v.as_u64()) {
|
|
163
|
+
self.messages.lock().unwrap().push(seq as usize);
|
|
164
|
+
Some(json!({"received": seq}))
|
|
165
|
+
} else {
|
|
166
|
+
None
|
|
167
|
+
}
|
|
168
|
+
}
|
|
169
|
+
}
|
|
170
|
+
|
|
171
|
+
/// Handler that simulates high-load buffering behavior
|
|
172
|
+
#[derive(Debug)]
|
|
173
|
+
struct BufferingHandler {
|
|
174
|
+
processed: Arc<AtomicUsize>,
|
|
175
|
+
}
|
|
176
|
+
|
|
177
|
+
impl BufferingHandler {
|
|
178
|
+
fn new() -> Self {
|
|
179
|
+
Self {
|
|
180
|
+
processed: Arc::new(AtomicUsize::new(0)),
|
|
181
|
+
}
|
|
182
|
+
}
|
|
183
|
+
}
|
|
184
|
+
|
|
185
|
+
impl WebSocketHandler for BufferingHandler {
|
|
186
|
+
async fn handle_message(&self, message: Value) -> Option<Value> {
|
|
187
|
+
self.processed.fetch_add(1, Ordering::SeqCst);
|
|
188
|
+
// Simulate processing delay
|
|
189
|
+
sleep(Duration::from_millis(1)).await;
|
|
190
|
+
Some(json!({"processed": message, "total": self.processed.load(Ordering::SeqCst)}))
|
|
191
|
+
}
|
|
192
|
+
}
|
|
193
|
+
|
|
194
|
+
// ===== Test 1: WebSocket Connection Lifecycle =====
|
|
195
|
+
|
|
196
|
+
#[tokio::test]
|
|
197
|
+
async fn test_websocket_connection_initialization() {
|
|
198
|
+
let handler = LifecycleHandler::new();
|
|
199
|
+
|
|
200
|
+
// Initially, no events should have fired
|
|
201
|
+
assert!(!handler.connect_called.load(Ordering::SeqCst));
|
|
202
|
+
assert!(!handler.disconnect_called.load(Ordering::SeqCst));
|
|
203
|
+
assert_eq!(handler.message_count.load(Ordering::SeqCst), 0);
|
|
204
|
+
|
|
205
|
+
// Verify state can be created
|
|
206
|
+
let _state: WebSocketState<LifecycleHandler> = WebSocketState::new(handler.clone());
|
|
207
|
+
}
|
|
208
|
+
|
|
209
|
+
#[tokio::test]
|
|
210
|
+
async fn test_websocket_connection_lifecycle_state_transitions() {
|
|
211
|
+
let handler = LifecycleHandler::new();
|
|
212
|
+
handler.reset();
|
|
213
|
+
|
|
214
|
+
// Simulate connection lifecycle
|
|
215
|
+
handler.on_connect().await;
|
|
216
|
+
assert!(handler.connect_called.load(Ordering::SeqCst));
|
|
217
|
+
assert!(!handler.disconnect_called.load(Ordering::SeqCst));
|
|
218
|
+
|
|
219
|
+
// Process some messages
|
|
220
|
+
let msg = json!({"test": "data"});
|
|
221
|
+
let _resp = handler.handle_message(msg).await;
|
|
222
|
+
|
|
223
|
+
// Simulate disconnection
|
|
224
|
+
handler.on_disconnect().await;
|
|
225
|
+
assert!(handler.connect_called.load(Ordering::SeqCst));
|
|
226
|
+
assert!(handler.disconnect_called.load(Ordering::SeqCst));
|
|
227
|
+
}
|
|
228
|
+
|
|
229
|
+
#[tokio::test]
|
|
230
|
+
async fn test_websocket_sends_and_receives_single_message() {
|
|
231
|
+
let handler = EchoHandler;
|
|
232
|
+
let msg = json!({"test": "message"});
|
|
233
|
+
let response = handler.handle_message(msg.clone()).await;
|
|
234
|
+
|
|
235
|
+
assert_eq!(response, Some(msg));
|
|
236
|
+
}
|
|
237
|
+
|
|
238
|
+
// ===== Test 2: Concurrent Messages from Multiple Clients =====
|
|
239
|
+
|
|
240
|
+
#[tokio::test]
|
|
241
|
+
async fn test_multiple_messages_from_same_connection() {
|
|
242
|
+
let handler = LifecycleHandler::new();
|
|
243
|
+
handler.reset();
|
|
244
|
+
|
|
245
|
+
let msg1 = json!({"id": 1, "text": "first"});
|
|
246
|
+
let msg2 = json!({"id": 2, "text": "second"});
|
|
247
|
+
let msg3 = json!({"id": 3, "text": "third"});
|
|
248
|
+
|
|
249
|
+
let _resp1 = handler.handle_message(msg1).await;
|
|
250
|
+
let _resp2 = handler.handle_message(msg2).await;
|
|
251
|
+
let _resp3 = handler.handle_message(msg3).await;
|
|
252
|
+
|
|
253
|
+
assert_eq!(handler.message_count.load(Ordering::SeqCst), 3);
|
|
254
|
+
}
|
|
255
|
+
|
|
256
|
+
#[tokio::test]
|
|
257
|
+
async fn test_concurrent_message_processing() {
|
|
258
|
+
let handler = Arc::new(LifecycleHandler::new());
|
|
259
|
+
|
|
260
|
+
// Simulate concurrent message processing
|
|
261
|
+
let mut handles = vec![];
|
|
262
|
+
|
|
263
|
+
for i in 0..10 {
|
|
264
|
+
let handler_clone = handler.clone();
|
|
265
|
+
let handle = tokio::spawn(async move {
|
|
266
|
+
let msg = json!({"id": i, "data": format!("message_{}", i)});
|
|
267
|
+
handler_clone.handle_message(msg).await
|
|
268
|
+
});
|
|
269
|
+
handles.push(handle);
|
|
270
|
+
}
|
|
271
|
+
|
|
272
|
+
// Wait for all to complete
|
|
273
|
+
for handle in handles {
|
|
274
|
+
let _ = handle.await;
|
|
275
|
+
}
|
|
276
|
+
|
|
277
|
+
assert_eq!(handler.message_count.load(Ordering::SeqCst), 10);
|
|
278
|
+
assert_eq!(handler.messages.lock().unwrap().len(), 10);
|
|
279
|
+
}
|
|
280
|
+
|
|
281
|
+
#[tokio::test]
|
|
282
|
+
async fn test_multiple_concurrent_connections_isolation() {
|
|
283
|
+
let handler1 = LifecycleHandler::new();
|
|
284
|
+
let handler2 = LifecycleHandler::new();
|
|
285
|
+
|
|
286
|
+
handler1.reset();
|
|
287
|
+
handler2.reset();
|
|
288
|
+
|
|
289
|
+
// Simulate messages from two different connections
|
|
290
|
+
let msg1 = json!({"connection": 1, "seq": 1});
|
|
291
|
+
let msg2 = json!({"connection": 2, "seq": 1});
|
|
292
|
+
|
|
293
|
+
let _resp1 = handler1.handle_message(msg1).await;
|
|
294
|
+
let _resp2 = handler2.handle_message(msg2).await;
|
|
295
|
+
|
|
296
|
+
assert_eq!(handler1.message_count.load(Ordering::SeqCst), 1);
|
|
297
|
+
assert_eq!(handler2.message_count.load(Ordering::SeqCst), 1);
|
|
298
|
+
|
|
299
|
+
// Ensure each handler only got its own messages
|
|
300
|
+
assert_eq!(handler1.messages.lock().unwrap()[0].get("connection").unwrap(), 1);
|
|
301
|
+
assert_eq!(handler2.messages.lock().unwrap()[0].get("connection").unwrap(), 2);
|
|
302
|
+
}
|
|
303
|
+
|
|
304
|
+
// ===== Test 3: Message Ordering Guarantees =====
|
|
305
|
+
|
|
306
|
+
#[tokio::test]
|
|
307
|
+
async fn test_message_ordering_sequential_delivery() {
|
|
308
|
+
let handler = OrderingHandler::new();
|
|
309
|
+
|
|
310
|
+
// Send messages with sequence numbers
|
|
311
|
+
for seq in 0..10 {
|
|
312
|
+
let msg = json!({"sequence": seq});
|
|
313
|
+
let _response = handler.handle_message(msg).await;
|
|
314
|
+
}
|
|
315
|
+
|
|
316
|
+
let messages = handler.messages.lock().unwrap();
|
|
317
|
+
let expected: Vec<usize> = (0..10).collect();
|
|
318
|
+
|
|
319
|
+
assert_eq!(*messages, expected);
|
|
320
|
+
}
|
|
321
|
+
|
|
322
|
+
#[tokio::test]
|
|
323
|
+
async fn test_message_ordering_concurrent_arrival() {
|
|
324
|
+
let handler = Arc::new(OrderingHandler::new());
|
|
325
|
+
|
|
326
|
+
// Send messages concurrently
|
|
327
|
+
let mut handles = vec![];
|
|
328
|
+
for seq in 0..20 {
|
|
329
|
+
let handler_clone = handler.clone();
|
|
330
|
+
let handle = tokio::spawn(async move {
|
|
331
|
+
let msg = json!({"sequence": seq});
|
|
332
|
+
handler_clone.handle_message(msg).await
|
|
333
|
+
});
|
|
334
|
+
handles.push(handle);
|
|
335
|
+
}
|
|
336
|
+
|
|
337
|
+
// Wait for all to complete
|
|
338
|
+
for handle in handles {
|
|
339
|
+
let _ = handle.await;
|
|
340
|
+
}
|
|
341
|
+
|
|
342
|
+
let messages = handler.messages.lock().unwrap();
|
|
343
|
+
assert_eq!(messages.len(), 20);
|
|
344
|
+
|
|
345
|
+
// Verify all sequence numbers are present (order may vary in concurrent)
|
|
346
|
+
let mut sorted = messages.clone();
|
|
347
|
+
sorted.sort();
|
|
348
|
+
let expected: Vec<usize> = (0..20).collect();
|
|
349
|
+
assert_eq!(sorted, expected);
|
|
350
|
+
}
|
|
351
|
+
|
|
352
|
+
#[tokio::test]
|
|
353
|
+
async fn test_message_ordering_with_delays() {
|
|
354
|
+
let handler = OrderingHandler::new();
|
|
355
|
+
|
|
356
|
+
// Send messages with small delays between them
|
|
357
|
+
for seq in 0..5 {
|
|
358
|
+
let msg = json!({"sequence": seq});
|
|
359
|
+
let _response = handler.handle_message(msg).await;
|
|
360
|
+
sleep(Duration::from_millis(1)).await;
|
|
361
|
+
}
|
|
362
|
+
|
|
363
|
+
let messages = handler.messages.lock().unwrap();
|
|
364
|
+
let expected: Vec<usize> = (0..5).collect();
|
|
365
|
+
|
|
366
|
+
assert_eq!(*messages, expected);
|
|
367
|
+
}
|
|
368
|
+
|
|
369
|
+
// ===== Test 4: Connection Abort Handling =====
|
|
370
|
+
|
|
371
|
+
#[tokio::test]
|
|
372
|
+
async fn test_handler_disconnect_on_normal_close() {
|
|
373
|
+
let handler = LifecycleHandler::new();
|
|
374
|
+
handler.reset();
|
|
375
|
+
|
|
376
|
+
// Simulate normal disconnect
|
|
377
|
+
handler.on_disconnect().await;
|
|
378
|
+
|
|
379
|
+
assert!(handler.disconnect_called.load(Ordering::SeqCst));
|
|
380
|
+
}
|
|
381
|
+
|
|
382
|
+
#[tokio::test]
|
|
383
|
+
async fn test_handler_continues_after_failed_message() {
|
|
384
|
+
let handler = LifecycleHandler::new();
|
|
385
|
+
handler.reset();
|
|
386
|
+
|
|
387
|
+
let valid_msg = json!({"data": "test"});
|
|
388
|
+
let resp1 = handler.handle_message(valid_msg).await;
|
|
389
|
+
assert!(resp1.is_some());
|
|
390
|
+
|
|
391
|
+
// Handler should continue working after processing
|
|
392
|
+
let another_msg = json!({"data": "another"});
|
|
393
|
+
let resp2 = handler.handle_message(another_msg).await;
|
|
394
|
+
assert!(resp2.is_some());
|
|
395
|
+
|
|
396
|
+
assert_eq!(handler.message_count.load(Ordering::SeqCst), 2);
|
|
397
|
+
}
|
|
398
|
+
|
|
399
|
+
#[tokio::test]
|
|
400
|
+
async fn test_handler_state_after_error() {
|
|
401
|
+
let handler = ErrorHandler::new();
|
|
402
|
+
|
|
403
|
+
// Set handler to error mode
|
|
404
|
+
handler.should_error.store(true, Ordering::SeqCst);
|
|
405
|
+
|
|
406
|
+
let msg1 = json!({"test": 1});
|
|
407
|
+
let resp1 = handler.handle_message(msg1).await;
|
|
408
|
+
|
|
409
|
+
// Error response should be None
|
|
410
|
+
assert!(resp1.is_none());
|
|
411
|
+
assert_eq!(handler.error_count.load(Ordering::SeqCst), 1);
|
|
412
|
+
|
|
413
|
+
// Disable error mode and verify handler recovers
|
|
414
|
+
handler.should_error.store(false, Ordering::SeqCst);
|
|
415
|
+
|
|
416
|
+
let msg2 = json!({"test": 2});
|
|
417
|
+
let resp2 = handler.handle_message(msg2).await;
|
|
418
|
+
|
|
419
|
+
// Should now respond normally
|
|
420
|
+
assert!(resp2.is_some());
|
|
421
|
+
assert_eq!(handler.error_count.load(Ordering::SeqCst), 1);
|
|
422
|
+
}
|
|
423
|
+
|
|
424
|
+
// ===== Test 5: Schema Validation Rejection =====
|
|
425
|
+
|
|
426
|
+
#[tokio::test]
|
|
427
|
+
async fn test_schema_validation_accepts_valid_message() {
|
|
428
|
+
let handler = SchemaValidationHandler::new();
|
|
429
|
+
|
|
430
|
+
let valid_msg = json!({"type": "test", "data": "content"});
|
|
431
|
+
let response = handler.handle_message(valid_msg).await;
|
|
432
|
+
|
|
433
|
+
assert!(response.is_some());
|
|
434
|
+
let resp = response.unwrap();
|
|
435
|
+
assert_eq!(resp.get("status").unwrap(), "ok");
|
|
436
|
+
}
|
|
437
|
+
|
|
438
|
+
#[tokio::test]
|
|
439
|
+
async fn test_schema_validation_rejects_invalid_message() {
|
|
440
|
+
let handler = SchemaValidationHandler::new();
|
|
441
|
+
|
|
442
|
+
// Message missing required "type" field
|
|
443
|
+
let invalid_msg = json!({"data": "content"});
|
|
444
|
+
let response = handler.handle_message(invalid_msg.clone()).await;
|
|
445
|
+
|
|
446
|
+
// Handler returns None for invalid messages
|
|
447
|
+
assert!(response.is_none());
|
|
448
|
+
|
|
449
|
+
// Verify invalid message was recorded
|
|
450
|
+
assert_eq!(handler.invalid_messages.lock().unwrap().len(), 1);
|
|
451
|
+
}
|
|
452
|
+
|
|
453
|
+
#[tokio::test]
|
|
454
|
+
async fn test_schema_validation_multiple_validations() {
|
|
455
|
+
let handler = SchemaValidationHandler::new();
|
|
456
|
+
|
|
457
|
+
let valid1 = json!({"type": "cmd", "action": "start"});
|
|
458
|
+
let invalid1 = json!({"action": "start"}); // missing type
|
|
459
|
+
let valid2 = json!({"type": "query", "params": {}});
|
|
460
|
+
let invalid2 = json!({"id": 123}); // missing type
|
|
461
|
+
|
|
462
|
+
let r1 = handler.handle_message(valid1).await;
|
|
463
|
+
let r2 = handler.handle_message(invalid1).await;
|
|
464
|
+
let r3 = handler.handle_message(valid2).await;
|
|
465
|
+
let r4 = handler.handle_message(invalid2).await;
|
|
466
|
+
|
|
467
|
+
assert!(r1.is_some());
|
|
468
|
+
assert!(r2.is_none());
|
|
469
|
+
assert!(r3.is_some());
|
|
470
|
+
assert!(r4.is_none());
|
|
471
|
+
|
|
472
|
+
assert_eq!(handler.invalid_messages.lock().unwrap().len(), 2);
|
|
473
|
+
}
|
|
474
|
+
|
|
475
|
+
#[tokio::test]
|
|
476
|
+
async fn test_schema_validation_type_checking() {
|
|
477
|
+
let handler = SchemaValidationHandler::new();
|
|
478
|
+
|
|
479
|
+
// Message with type field but wrong type
|
|
480
|
+
let msg_with_number_type = json!({"type": 123});
|
|
481
|
+
let response = handler.handle_message(msg_with_number_type).await;
|
|
482
|
+
|
|
483
|
+
// Should accept because we only check for presence of "type" field
|
|
484
|
+
assert!(response.is_some());
|
|
485
|
+
}
|
|
486
|
+
|
|
487
|
+
// ===== Test 6: Handler Error Propagation =====
|
|
488
|
+
|
|
489
|
+
#[tokio::test]
|
|
490
|
+
async fn test_handler_error_state_preservation() {
|
|
491
|
+
let handler = ErrorHandler::new();
|
|
492
|
+
|
|
493
|
+
handler.should_error.store(true, Ordering::SeqCst);
|
|
494
|
+
|
|
495
|
+
// Send multiple messages while in error state
|
|
496
|
+
for i in 0..5 {
|
|
497
|
+
let msg = json!({"id": i});
|
|
498
|
+
let _resp = handler.handle_message(msg).await;
|
|
499
|
+
}
|
|
500
|
+
|
|
501
|
+
// All should have failed
|
|
502
|
+
assert_eq!(handler.error_count.load(Ordering::SeqCst), 5);
|
|
503
|
+
}
|
|
504
|
+
|
|
505
|
+
#[tokio::test]
|
|
506
|
+
async fn test_handler_error_recovery_transitions() {
|
|
507
|
+
let handler = ErrorHandler::new();
|
|
508
|
+
|
|
509
|
+
// Normal operation
|
|
510
|
+
let msg1 = json!({"id": 1});
|
|
511
|
+
let resp1 = handler.handle_message(msg1).await;
|
|
512
|
+
assert!(resp1.is_some());
|
|
513
|
+
|
|
514
|
+
// Transition to error state
|
|
515
|
+
handler.should_error.store(true, Ordering::SeqCst);
|
|
516
|
+
let msg2 = json!({"id": 2});
|
|
517
|
+
let resp2 = handler.handle_message(msg2).await;
|
|
518
|
+
assert!(resp2.is_none());
|
|
519
|
+
|
|
520
|
+
// Transition back to normal
|
|
521
|
+
handler.should_error.store(false, Ordering::SeqCst);
|
|
522
|
+
let msg3 = json!({"id": 3});
|
|
523
|
+
let resp3 = handler.handle_message(msg3).await;
|
|
524
|
+
assert!(resp3.is_some());
|
|
525
|
+
}
|
|
526
|
+
|
|
527
|
+
#[tokio::test]
|
|
528
|
+
async fn test_selective_error_handling() {
|
|
529
|
+
let handler = ErrorHandler::new();
|
|
530
|
+
|
|
531
|
+
// Normal then error
|
|
532
|
+
handler.should_error.store(false, Ordering::SeqCst);
|
|
533
|
+
let msg1 = json!({"id": 1});
|
|
534
|
+
let resp1 = handler.handle_message(msg1).await;
|
|
535
|
+
assert!(resp1.is_some());
|
|
536
|
+
|
|
537
|
+
// Error state
|
|
538
|
+
handler.should_error.store(true, Ordering::SeqCst);
|
|
539
|
+
let msg2 = json!({"id": 2});
|
|
540
|
+
let resp2 = handler.handle_message(msg2).await;
|
|
541
|
+
assert!(resp2.is_none());
|
|
542
|
+
|
|
543
|
+
// Another normal
|
|
544
|
+
handler.should_error.store(false, Ordering::SeqCst);
|
|
545
|
+
let msg3 = json!({"id": 3});
|
|
546
|
+
let resp3 = handler.handle_message(msg3).await;
|
|
547
|
+
assert!(resp3.is_some());
|
|
548
|
+
|
|
549
|
+
assert_eq!(handler.error_count.load(Ordering::SeqCst), 1);
|
|
550
|
+
}
|
|
551
|
+
|
|
552
|
+
// ===== Test 7: Message Buffering Under High Load =====
|
|
553
|
+
|
|
554
|
+
#[tokio::test]
|
|
555
|
+
async fn test_message_buffering_rapid_succession() {
|
|
556
|
+
let handler = BufferingHandler::new();
|
|
557
|
+
|
|
558
|
+
// Send messages rapidly
|
|
559
|
+
for i in 0..50 {
|
|
560
|
+
let msg = json!({"id": i, "timestamp": "2024-01-01T00:00:00Z"});
|
|
561
|
+
let _response = handler.handle_message(msg).await;
|
|
562
|
+
}
|
|
563
|
+
|
|
564
|
+
assert_eq!(handler.processed.load(Ordering::SeqCst), 50);
|
|
565
|
+
}
|
|
566
|
+
|
|
567
|
+
#[tokio::test]
|
|
568
|
+
async fn test_message_buffering_concurrent_load() {
|
|
569
|
+
let handler = Arc::new(BufferingHandler::new());
|
|
570
|
+
|
|
571
|
+
let mut handles = vec![];
|
|
572
|
+
|
|
573
|
+
// Spawn 10 concurrent tasks, each sending 10 messages
|
|
574
|
+
for task_id in 0..10 {
|
|
575
|
+
let handler_clone = handler.clone();
|
|
576
|
+
let handle = tokio::spawn(async move {
|
|
577
|
+
for seq in 0..10 {
|
|
578
|
+
let msg = json!({
|
|
579
|
+
"task": task_id,
|
|
580
|
+
"sequence": seq,
|
|
581
|
+
"data": format!("task_{}_msg_{}", task_id, seq)
|
|
582
|
+
});
|
|
583
|
+
let _resp = handler_clone.handle_message(msg).await;
|
|
584
|
+
}
|
|
585
|
+
});
|
|
586
|
+
handles.push(handle);
|
|
587
|
+
}
|
|
588
|
+
|
|
589
|
+
for handle in handles {
|
|
590
|
+
let _ = handle.await;
|
|
591
|
+
}
|
|
592
|
+
|
|
593
|
+
// Should have processed all messages from all tasks
|
|
594
|
+
assert_eq!(handler.processed.load(Ordering::SeqCst), 100);
|
|
595
|
+
}
|
|
596
|
+
|
|
597
|
+
#[tokio::test]
|
|
598
|
+
async fn test_message_buffering_response_correctness_under_load() {
|
|
599
|
+
let handler = BufferingHandler::new();
|
|
600
|
+
|
|
601
|
+
// Send burst of messages and verify responses are sensible
|
|
602
|
+
for i in 0..20 {
|
|
603
|
+
let msg = json!({"burst_id": i, "data": "test"});
|
|
604
|
+
let response = handler.handle_message(msg.clone()).await;
|
|
605
|
+
|
|
606
|
+
// Verify response structure
|
|
607
|
+
assert!(response.is_some());
|
|
608
|
+
let resp = response.unwrap();
|
|
609
|
+
assert!(resp.get("processed").is_some());
|
|
610
|
+
assert!(resp.get("total").is_some());
|
|
611
|
+
}
|
|
612
|
+
|
|
613
|
+
assert_eq!(handler.processed.load(Ordering::SeqCst), 20);
|
|
614
|
+
}
|
|
615
|
+
|
|
616
|
+
#[tokio::test]
|
|
617
|
+
async fn test_message_buffering_maintains_order_under_load() {
|
|
618
|
+
let handler = Arc::new(OrderingHandler::new());
|
|
619
|
+
|
|
620
|
+
let mut handles = vec![];
|
|
621
|
+
|
|
622
|
+
// Send 100 messages from a single task to ensure ordering
|
|
623
|
+
let handler_clone = handler.clone();
|
|
624
|
+
let handle = tokio::spawn(async move {
|
|
625
|
+
for seq in 0..100 {
|
|
626
|
+
let msg = json!({"sequence": seq});
|
|
627
|
+
let _resp = handler_clone.handle_message(msg).await;
|
|
628
|
+
}
|
|
629
|
+
});
|
|
630
|
+
handles.push(handle);
|
|
631
|
+
|
|
632
|
+
for handle in handles {
|
|
633
|
+
let _ = handle.await;
|
|
634
|
+
}
|
|
635
|
+
|
|
636
|
+
let messages = handler.messages.lock().unwrap();
|
|
637
|
+
let expected: Vec<usize> = (0..100).collect();
|
|
638
|
+
assert_eq!(*messages, expected);
|
|
639
|
+
}
|
|
640
|
+
|
|
641
|
+
// ===== Additional Edge Cases =====
|
|
642
|
+
|
|
643
|
+
#[tokio::test]
|
|
644
|
+
async fn test_large_payload_handling() {
|
|
645
|
+
let handler = EchoHandler;
|
|
646
|
+
|
|
647
|
+
// Create a large JSON payload
|
|
648
|
+
let large_array: Vec<i32> = (0..1000).collect();
|
|
649
|
+
let large_msg = json!({
|
|
650
|
+
"type": "large_payload",
|
|
651
|
+
"data": large_array,
|
|
652
|
+
"metadata": {
|
|
653
|
+
"size": 1000,
|
|
654
|
+
"description": "This is a test of large payload handling"
|
|
655
|
+
}
|
|
656
|
+
});
|
|
657
|
+
|
|
658
|
+
let response = handler.handle_message(large_msg.clone()).await;
|
|
659
|
+
assert!(response.is_some());
|
|
660
|
+
assert_eq!(response.unwrap(), large_msg);
|
|
661
|
+
}
|
|
662
|
+
|
|
663
|
+
#[tokio::test]
|
|
664
|
+
async fn test_deeply_nested_message_handling() {
|
|
665
|
+
let handler = EchoHandler;
|
|
666
|
+
|
|
667
|
+
// Create deeply nested JSON
|
|
668
|
+
let mut nested = json!({"value": "deep"});
|
|
669
|
+
for _ in 0..50 {
|
|
670
|
+
nested = json!({"level": nested});
|
|
671
|
+
}
|
|
672
|
+
|
|
673
|
+
let response = handler.handle_message(nested.clone()).await;
|
|
674
|
+
assert!(response.is_some());
|
|
675
|
+
assert_eq!(response.unwrap(), nested);
|
|
676
|
+
}
|
|
677
|
+
|
|
678
|
+
#[tokio::test]
|
|
679
|
+
async fn test_unicode_and_special_characters() {
|
|
680
|
+
let handler = EchoHandler;
|
|
681
|
+
|
|
682
|
+
let unicode_msg = json!({
|
|
683
|
+
"emoji": "🚀💡🔥",
|
|
684
|
+
"chinese": "ä½ å¥½ä¸–ç•Œ",
|
|
685
|
+
"arabic": "Ù…Ø±ØØ¨Ø§ بالعالم",
|
|
686
|
+
"special": "!@#$%^&*()",
|
|
687
|
+
"newlines": "line1\nline2\nline3"
|
|
688
|
+
});
|
|
689
|
+
|
|
690
|
+
let response = handler.handle_message(unicode_msg.clone()).await;
|
|
691
|
+
assert!(response.is_some());
|
|
692
|
+
assert_eq!(response.unwrap(), unicode_msg);
|
|
693
|
+
}
|
|
694
|
+
|
|
695
|
+
#[tokio::test]
|
|
696
|
+
async fn test_null_and_empty_values() {
|
|
697
|
+
let handler = EchoHandler;
|
|
698
|
+
|
|
699
|
+
let test_cases = vec![
|
|
700
|
+
json!({"value": null}),
|
|
701
|
+
json!({"array": []}),
|
|
702
|
+
json!({"object": {}}),
|
|
703
|
+
json!({"string": ""}),
|
|
704
|
+
];
|
|
705
|
+
|
|
706
|
+
for msg in test_cases {
|
|
707
|
+
let response = handler.handle_message(msg.clone()).await;
|
|
708
|
+
assert!(response.is_some());
|
|
709
|
+
assert_eq!(response.unwrap(), msg);
|
|
710
|
+
}
|
|
711
|
+
}
|
|
712
|
+
|
|
713
|
+
#[tokio::test]
|
|
714
|
+
async fn test_mixed_type_arrays() {
|
|
715
|
+
let handler = EchoHandler;
|
|
716
|
+
|
|
717
|
+
let msg = json!({
|
|
718
|
+
"mixed": [1, "two", 3.0, true, null, {"key": "value"}, []]
|
|
719
|
+
});
|
|
720
|
+
|
|
721
|
+
let response = handler.handle_message(msg.clone()).await;
|
|
722
|
+
assert!(response.is_some());
|
|
723
|
+
assert_eq!(response.unwrap(), msg);
|
|
724
|
+
}
|