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.
Files changed (138) hide show
  1. checksums.yaml +7 -0
  2. data/LICENSE +1 -0
  3. data/README.md +659 -0
  4. data/ext/spikard_rb/Cargo.toml +17 -0
  5. data/ext/spikard_rb/extconf.rb +10 -0
  6. data/ext/spikard_rb/src/lib.rs +6 -0
  7. data/lib/spikard/app.rb +405 -0
  8. data/lib/spikard/background.rb +27 -0
  9. data/lib/spikard/config.rb +396 -0
  10. data/lib/spikard/converters.rb +13 -0
  11. data/lib/spikard/handler_wrapper.rb +113 -0
  12. data/lib/spikard/provide.rb +214 -0
  13. data/lib/spikard/response.rb +173 -0
  14. data/lib/spikard/schema.rb +243 -0
  15. data/lib/spikard/sse.rb +111 -0
  16. data/lib/spikard/streaming_response.rb +44 -0
  17. data/lib/spikard/testing.rb +221 -0
  18. data/lib/spikard/upload_file.rb +131 -0
  19. data/lib/spikard/version.rb +5 -0
  20. data/lib/spikard/websocket.rb +59 -0
  21. data/lib/spikard.rb +43 -0
  22. data/sig/spikard.rbs +366 -0
  23. data/vendor/bundle/ruby/3.4.0/gems/diff-lcs-1.6.2/mise.toml +5 -0
  24. data/vendor/bundle/ruby/3.4.0/gems/rake-compiler-dock-1.10.0/build/buildkitd.toml +2 -0
  25. data/vendor/crates/spikard-bindings-shared/Cargo.toml +63 -0
  26. data/vendor/crates/spikard-bindings-shared/examples/config_extraction.rs +139 -0
  27. data/vendor/crates/spikard-bindings-shared/src/config_extractor.rs +561 -0
  28. data/vendor/crates/spikard-bindings-shared/src/conversion_traits.rs +194 -0
  29. data/vendor/crates/spikard-bindings-shared/src/di_traits.rs +246 -0
  30. data/vendor/crates/spikard-bindings-shared/src/error_response.rs +403 -0
  31. data/vendor/crates/spikard-bindings-shared/src/handler_base.rs +274 -0
  32. data/vendor/crates/spikard-bindings-shared/src/lib.rs +25 -0
  33. data/vendor/crates/spikard-bindings-shared/src/lifecycle_base.rs +298 -0
  34. data/vendor/crates/spikard-bindings-shared/src/lifecycle_executor.rs +637 -0
  35. data/vendor/crates/spikard-bindings-shared/src/response_builder.rs +309 -0
  36. data/vendor/crates/spikard-bindings-shared/src/test_client_base.rs +248 -0
  37. data/vendor/crates/spikard-bindings-shared/src/validation_helpers.rs +355 -0
  38. data/vendor/crates/spikard-bindings-shared/tests/comprehensive_coverage.rs +502 -0
  39. data/vendor/crates/spikard-bindings-shared/tests/error_response_edge_cases.rs +389 -0
  40. data/vendor/crates/spikard-bindings-shared/tests/handler_base_integration.rs +413 -0
  41. data/vendor/crates/spikard-core/Cargo.toml +40 -0
  42. data/vendor/crates/spikard-core/src/bindings/mod.rs +3 -0
  43. data/vendor/crates/spikard-core/src/bindings/response.rs +133 -0
  44. data/vendor/crates/spikard-core/src/debug.rs +63 -0
  45. data/vendor/crates/spikard-core/src/di/container.rs +726 -0
  46. data/vendor/crates/spikard-core/src/di/dependency.rs +273 -0
  47. data/vendor/crates/spikard-core/src/di/error.rs +118 -0
  48. data/vendor/crates/spikard-core/src/di/factory.rs +538 -0
  49. data/vendor/crates/spikard-core/src/di/graph.rs +545 -0
  50. data/vendor/crates/spikard-core/src/di/mod.rs +192 -0
  51. data/vendor/crates/spikard-core/src/di/resolved.rs +411 -0
  52. data/vendor/crates/spikard-core/src/di/value.rs +283 -0
  53. data/vendor/crates/spikard-core/src/errors.rs +39 -0
  54. data/vendor/crates/spikard-core/src/http.rs +153 -0
  55. data/vendor/crates/spikard-core/src/lib.rs +29 -0
  56. data/vendor/crates/spikard-core/src/lifecycle.rs +422 -0
  57. data/vendor/crates/spikard-core/src/metadata.rs +397 -0
  58. data/vendor/crates/spikard-core/src/parameters.rs +723 -0
  59. data/vendor/crates/spikard-core/src/problem.rs +310 -0
  60. data/vendor/crates/spikard-core/src/request_data.rs +189 -0
  61. data/vendor/crates/spikard-core/src/router.rs +249 -0
  62. data/vendor/crates/spikard-core/src/schema_registry.rs +183 -0
  63. data/vendor/crates/spikard-core/src/type_hints.rs +304 -0
  64. data/vendor/crates/spikard-core/src/validation/error_mapper.rs +689 -0
  65. data/vendor/crates/spikard-core/src/validation/mod.rs +459 -0
  66. data/vendor/crates/spikard-http/Cargo.toml +58 -0
  67. data/vendor/crates/spikard-http/examples/sse-notifications.rs +147 -0
  68. data/vendor/crates/spikard-http/examples/websocket-chat.rs +91 -0
  69. data/vendor/crates/spikard-http/src/auth.rs +247 -0
  70. data/vendor/crates/spikard-http/src/background.rs +1562 -0
  71. data/vendor/crates/spikard-http/src/bindings/mod.rs +3 -0
  72. data/vendor/crates/spikard-http/src/bindings/response.rs +1 -0
  73. data/vendor/crates/spikard-http/src/body_metadata.rs +8 -0
  74. data/vendor/crates/spikard-http/src/cors.rs +490 -0
  75. data/vendor/crates/spikard-http/src/debug.rs +63 -0
  76. data/vendor/crates/spikard-http/src/di_handler.rs +1878 -0
  77. data/vendor/crates/spikard-http/src/handler_response.rs +532 -0
  78. data/vendor/crates/spikard-http/src/handler_trait.rs +861 -0
  79. data/vendor/crates/spikard-http/src/handler_trait_tests.rs +284 -0
  80. data/vendor/crates/spikard-http/src/lib.rs +524 -0
  81. data/vendor/crates/spikard-http/src/lifecycle/adapter.rs +149 -0
  82. data/vendor/crates/spikard-http/src/lifecycle.rs +428 -0
  83. data/vendor/crates/spikard-http/src/middleware/mod.rs +285 -0
  84. data/vendor/crates/spikard-http/src/middleware/multipart.rs +930 -0
  85. data/vendor/crates/spikard-http/src/middleware/urlencoded.rs +541 -0
  86. data/vendor/crates/spikard-http/src/middleware/validation.rs +287 -0
  87. data/vendor/crates/spikard-http/src/openapi/mod.rs +309 -0
  88. data/vendor/crates/spikard-http/src/openapi/parameter_extraction.rs +535 -0
  89. data/vendor/crates/spikard-http/src/openapi/schema_conversion.rs +867 -0
  90. data/vendor/crates/spikard-http/src/openapi/spec_generation.rs +678 -0
  91. data/vendor/crates/spikard-http/src/query_parser.rs +369 -0
  92. data/vendor/crates/spikard-http/src/response.rs +399 -0
  93. data/vendor/crates/spikard-http/src/server/handler.rs +1557 -0
  94. data/vendor/crates/spikard-http/src/server/lifecycle_execution.rs +98 -0
  95. data/vendor/crates/spikard-http/src/server/mod.rs +806 -0
  96. data/vendor/crates/spikard-http/src/server/request_extraction.rs +630 -0
  97. data/vendor/crates/spikard-http/src/server/routing_factory.rs +497 -0
  98. data/vendor/crates/spikard-http/src/sse.rs +961 -0
  99. data/vendor/crates/spikard-http/src/testing/form.rs +14 -0
  100. data/vendor/crates/spikard-http/src/testing/multipart.rs +60 -0
  101. data/vendor/crates/spikard-http/src/testing/test_client.rs +285 -0
  102. data/vendor/crates/spikard-http/src/testing.rs +377 -0
  103. data/vendor/crates/spikard-http/src/websocket.rs +831 -0
  104. data/vendor/crates/spikard-http/tests/background_behavior.rs +918 -0
  105. data/vendor/crates/spikard-http/tests/common/handlers.rs +308 -0
  106. data/vendor/crates/spikard-http/tests/common/mod.rs +21 -0
  107. data/vendor/crates/spikard-http/tests/di_integration.rs +202 -0
  108. data/vendor/crates/spikard-http/tests/doc_snippets.rs +4 -0
  109. data/vendor/crates/spikard-http/tests/lifecycle_execution.rs +1135 -0
  110. data/vendor/crates/spikard-http/tests/multipart_behavior.rs +688 -0
  111. data/vendor/crates/spikard-http/tests/server_config_builder.rs +324 -0
  112. data/vendor/crates/spikard-http/tests/sse_behavior.rs +728 -0
  113. data/vendor/crates/spikard-http/tests/websocket_behavior.rs +724 -0
  114. data/vendor/crates/spikard-rb/Cargo.toml +43 -0
  115. data/vendor/crates/spikard-rb/build.rs +199 -0
  116. data/vendor/crates/spikard-rb/src/background.rs +63 -0
  117. data/vendor/crates/spikard-rb/src/config/mod.rs +5 -0
  118. data/vendor/crates/spikard-rb/src/config/server_config.rs +283 -0
  119. data/vendor/crates/spikard-rb/src/conversion.rs +459 -0
  120. data/vendor/crates/spikard-rb/src/di/builder.rs +105 -0
  121. data/vendor/crates/spikard-rb/src/di/mod.rs +413 -0
  122. data/vendor/crates/spikard-rb/src/handler.rs +612 -0
  123. data/vendor/crates/spikard-rb/src/integration/mod.rs +3 -0
  124. data/vendor/crates/spikard-rb/src/lib.rs +1857 -0
  125. data/vendor/crates/spikard-rb/src/lifecycle.rs +275 -0
  126. data/vendor/crates/spikard-rb/src/metadata/mod.rs +5 -0
  127. data/vendor/crates/spikard-rb/src/metadata/route_extraction.rs +427 -0
  128. data/vendor/crates/spikard-rb/src/runtime/mod.rs +5 -0
  129. data/vendor/crates/spikard-rb/src/runtime/server_runner.rs +326 -0
  130. data/vendor/crates/spikard-rb/src/server.rs +283 -0
  131. data/vendor/crates/spikard-rb/src/sse.rs +231 -0
  132. data/vendor/crates/spikard-rb/src/testing/client.rs +404 -0
  133. data/vendor/crates/spikard-rb/src/testing/mod.rs +7 -0
  134. data/vendor/crates/spikard-rb/src/testing/sse.rs +143 -0
  135. data/vendor/crates/spikard-rb/src/testing/websocket.rs +221 -0
  136. data/vendor/crates/spikard-rb/src/websocket.rs +233 -0
  137. data/vendor/crates/spikard-rb/tests/magnus_ffi_tests.rs +14 -0
  138. 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
+ }