spikard 0.3.6 → 0.6.1

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