spikard 0.5.0 → 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 (135) hide show
  1. checksums.yaml +4 -4
  2. data/LICENSE +1 -1
  3. data/README.md +674 -674
  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 -405
  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 -256
  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 -63
  24. data/vendor/crates/spikard-bindings-shared/examples/config_extraction.rs +132 -132
  25. data/vendor/crates/spikard-bindings-shared/src/config_extractor.rs +752 -752
  26. data/vendor/crates/spikard-bindings-shared/src/conversion_traits.rs +194 -194
  27. data/vendor/crates/spikard-bindings-shared/src/di_traits.rs +246 -246
  28. data/vendor/crates/spikard-bindings-shared/src/error_response.rs +401 -401
  29. data/vendor/crates/spikard-bindings-shared/src/handler_base.rs +238 -238
  30. data/vendor/crates/spikard-bindings-shared/src/lib.rs +24 -24
  31. data/vendor/crates/spikard-bindings-shared/src/lifecycle_base.rs +292 -292
  32. data/vendor/crates/spikard-bindings-shared/src/lifecycle_executor.rs +616 -616
  33. data/vendor/crates/spikard-bindings-shared/src/response_builder.rs +305 -305
  34. data/vendor/crates/spikard-bindings-shared/src/test_client_base.rs +248 -248
  35. data/vendor/crates/spikard-bindings-shared/src/validation_helpers.rs +351 -351
  36. data/vendor/crates/spikard-bindings-shared/tests/comprehensive_coverage.rs +454 -454
  37. data/vendor/crates/spikard-bindings-shared/tests/error_response_edge_cases.rs +383 -383
  38. data/vendor/crates/spikard-bindings-shared/tests/handler_base_integration.rs +280 -280
  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 -127
  43. data/vendor/crates/spikard-core/src/di/container.rs +702 -702
  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 -534
  47. data/vendor/crates/spikard-core/src/di/graph.rs +506 -506
  48. data/vendor/crates/spikard-core/src/di/mod.rs +192 -192
  49. data/vendor/crates/spikard-core/src/di/resolved.rs +405 -405
  50. data/vendor/crates/spikard-core/src/di/value.rs +281 -281
  51. data/vendor/crates/spikard-core/src/errors.rs +69 -69
  52. data/vendor/crates/spikard-core/src/http.rs +415 -415
  53. data/vendor/crates/spikard-core/src/lib.rs +29 -29
  54. data/vendor/crates/spikard-core/src/lifecycle.rs +1186 -1186
  55. data/vendor/crates/spikard-core/src/metadata.rs +389 -389
  56. data/vendor/crates/spikard-core/src/parameters.rs +2525 -2525
  57. data/vendor/crates/spikard-core/src/problem.rs +344 -344
  58. data/vendor/crates/spikard-core/src/request_data.rs +1154 -1154
  59. data/vendor/crates/spikard-core/src/router.rs +510 -510
  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 -688
  63. data/vendor/crates/spikard-core/src/validation/mod.rs +457 -457
  64. data/vendor/crates/spikard-http/Cargo.toml +62 -64
  65. data/vendor/crates/spikard-http/examples/sse-notifications.rs +148 -148
  66. data/vendor/crates/spikard-http/examples/websocket-chat.rs +92 -92
  67. data/vendor/crates/spikard-http/src/auth.rs +296 -296
  68. data/vendor/crates/spikard-http/src/background.rs +1860 -1860
  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 -1005
  73. data/vendor/crates/spikard-http/src/debug.rs +128 -128
  74. data/vendor/crates/spikard-http/src/di_handler.rs +1668 -1668
  75. data/vendor/crates/spikard-http/src/handler_response.rs +901 -901
  76. data/vendor/crates/spikard-http/src/handler_trait.rs +838 -830
  77. data/vendor/crates/spikard-http/src/handler_trait_tests.rs +290 -290
  78. data/vendor/crates/spikard-http/src/lib.rs +534 -534
  79. data/vendor/crates/spikard-http/src/lifecycle/adapter.rs +230 -230
  80. data/vendor/crates/spikard-http/src/lifecycle.rs +1193 -1193
  81. data/vendor/crates/spikard-http/src/middleware/mod.rs +560 -540
  82. data/vendor/crates/spikard-http/src/middleware/multipart.rs +912 -912
  83. data/vendor/crates/spikard-http/src/middleware/urlencoded.rs +513 -513
  84. data/vendor/crates/spikard-http/src/middleware/validation.rs +768 -735
  85. data/vendor/crates/spikard-http/src/openapi/mod.rs +309 -309
  86. data/vendor/crates/spikard-http/src/openapi/parameter_extraction.rs +535 -535
  87. data/vendor/crates/spikard-http/src/openapi/schema_conversion.rs +1363 -1363
  88. data/vendor/crates/spikard-http/src/openapi/spec_generation.rs +665 -665
  89. data/vendor/crates/spikard-http/src/query_parser.rs +793 -793
  90. data/vendor/crates/spikard-http/src/response.rs +720 -720
  91. data/vendor/crates/spikard-http/src/server/handler.rs +1650 -1650
  92. data/vendor/crates/spikard-http/src/server/lifecycle_execution.rs +234 -234
  93. data/vendor/crates/spikard-http/src/server/mod.rs +1593 -1502
  94. data/vendor/crates/spikard-http/src/server/request_extraction.rs +789 -770
  95. data/vendor/crates/spikard-http/src/server/routing_factory.rs +629 -599
  96. data/vendor/crates/spikard-http/src/sse.rs +1409 -1409
  97. data/vendor/crates/spikard-http/src/testing/form.rs +52 -52
  98. data/vendor/crates/spikard-http/src/testing/multipart.rs +64 -60
  99. data/vendor/crates/spikard-http/src/testing/test_client.rs +311 -283
  100. data/vendor/crates/spikard-http/src/testing.rs +406 -377
  101. data/vendor/crates/spikard-http/src/websocket.rs +1404 -1375
  102. data/vendor/crates/spikard-http/tests/background_behavior.rs +832 -832
  103. data/vendor/crates/spikard-http/tests/common/handlers.rs +309 -309
  104. data/vendor/crates/spikard-http/tests/common/mod.rs +26 -26
  105. data/vendor/crates/spikard-http/tests/di_integration.rs +192 -192
  106. data/vendor/crates/spikard-http/tests/doc_snippets.rs +5 -5
  107. data/vendor/crates/spikard-http/tests/lifecycle_execution.rs +1093 -1093
  108. data/vendor/crates/spikard-http/tests/multipart_behavior.rs +656 -656
  109. data/vendor/crates/spikard-http/tests/server_config_builder.rs +314 -314
  110. data/vendor/crates/spikard-http/tests/sse_behavior.rs +620 -620
  111. data/vendor/crates/spikard-http/tests/websocket_behavior.rs +663 -663
  112. data/vendor/crates/spikard-rb/Cargo.toml +48 -48
  113. data/vendor/crates/spikard-rb/build.rs +199 -199
  114. data/vendor/crates/spikard-rb/src/background.rs +63 -63
  115. data/vendor/crates/spikard-rb/src/config/mod.rs +5 -5
  116. data/vendor/crates/spikard-rb/src/config/server_config.rs +285 -285
  117. data/vendor/crates/spikard-rb/src/conversion.rs +554 -554
  118. data/vendor/crates/spikard-rb/src/di/builder.rs +100 -100
  119. data/vendor/crates/spikard-rb/src/di/mod.rs +375 -375
  120. data/vendor/crates/spikard-rb/src/handler.rs +618 -618
  121. data/vendor/crates/spikard-rb/src/integration/mod.rs +3 -3
  122. data/vendor/crates/spikard-rb/src/lib.rs +1806 -1810
  123. data/vendor/crates/spikard-rb/src/lifecycle.rs +275 -275
  124. data/vendor/crates/spikard-rb/src/metadata/mod.rs +5 -5
  125. data/vendor/crates/spikard-rb/src/metadata/route_extraction.rs +442 -447
  126. data/vendor/crates/spikard-rb/src/runtime/mod.rs +5 -5
  127. data/vendor/crates/spikard-rb/src/runtime/server_runner.rs +324 -324
  128. data/vendor/crates/spikard-rb/src/server.rs +305 -308
  129. data/vendor/crates/spikard-rb/src/sse.rs +231 -231
  130. data/vendor/crates/spikard-rb/src/testing/client.rs +538 -551
  131. data/vendor/crates/spikard-rb/src/testing/mod.rs +7 -7
  132. data/vendor/crates/spikard-rb/src/testing/sse.rs +143 -143
  133. data/vendor/crates/spikard-rb/src/testing/websocket.rs +608 -635
  134. data/vendor/crates/spikard-rb/src/websocket.rs +377 -374
  135. metadata +15 -1
@@ -1,663 +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
- }
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
+ }