spikard 0.3.6 → 0.5.0

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 (113) hide show
  1. checksums.yaml +4 -4
  2. data/README.md +21 -6
  3. data/ext/spikard_rb/Cargo.toml +2 -2
  4. data/lib/spikard/app.rb +33 -14
  5. data/lib/spikard/testing.rb +47 -12
  6. data/lib/spikard/version.rb +1 -1
  7. data/vendor/crates/spikard-bindings-shared/Cargo.toml +63 -0
  8. data/vendor/crates/spikard-bindings-shared/examples/config_extraction.rs +132 -0
  9. data/vendor/crates/spikard-bindings-shared/src/config_extractor.rs +752 -0
  10. data/vendor/crates/spikard-bindings-shared/src/conversion_traits.rs +194 -0
  11. data/vendor/crates/spikard-bindings-shared/src/di_traits.rs +246 -0
  12. data/vendor/crates/spikard-bindings-shared/src/error_response.rs +401 -0
  13. data/vendor/crates/spikard-bindings-shared/src/handler_base.rs +238 -0
  14. data/vendor/crates/spikard-bindings-shared/src/lib.rs +24 -0
  15. data/vendor/crates/spikard-bindings-shared/src/lifecycle_base.rs +292 -0
  16. data/vendor/crates/spikard-bindings-shared/src/lifecycle_executor.rs +616 -0
  17. data/vendor/crates/spikard-bindings-shared/src/response_builder.rs +305 -0
  18. data/vendor/crates/spikard-bindings-shared/src/test_client_base.rs +248 -0
  19. data/vendor/crates/spikard-bindings-shared/src/validation_helpers.rs +351 -0
  20. data/vendor/crates/spikard-bindings-shared/tests/comprehensive_coverage.rs +454 -0
  21. data/vendor/crates/spikard-bindings-shared/tests/error_response_edge_cases.rs +383 -0
  22. data/vendor/crates/spikard-bindings-shared/tests/handler_base_integration.rs +280 -0
  23. data/vendor/crates/spikard-core/Cargo.toml +4 -4
  24. data/vendor/crates/spikard-core/src/debug.rs +64 -0
  25. data/vendor/crates/spikard-core/src/di/container.rs +3 -27
  26. data/vendor/crates/spikard-core/src/di/factory.rs +1 -5
  27. data/vendor/crates/spikard-core/src/di/graph.rs +8 -47
  28. data/vendor/crates/spikard-core/src/di/mod.rs +1 -1
  29. data/vendor/crates/spikard-core/src/di/resolved.rs +1 -7
  30. data/vendor/crates/spikard-core/src/di/value.rs +2 -4
  31. data/vendor/crates/spikard-core/src/errors.rs +30 -0
  32. data/vendor/crates/spikard-core/src/http.rs +262 -0
  33. data/vendor/crates/spikard-core/src/lib.rs +1 -1
  34. data/vendor/crates/spikard-core/src/lifecycle.rs +764 -0
  35. data/vendor/crates/spikard-core/src/metadata.rs +389 -0
  36. data/vendor/crates/spikard-core/src/parameters.rs +1962 -159
  37. data/vendor/crates/spikard-core/src/problem.rs +34 -0
  38. data/vendor/crates/spikard-core/src/request_data.rs +966 -1
  39. data/vendor/crates/spikard-core/src/router.rs +263 -2
  40. data/vendor/crates/spikard-core/src/validation/error_mapper.rs +688 -0
  41. data/vendor/crates/spikard-core/src/{validation.rs → validation/mod.rs} +26 -268
  42. data/vendor/crates/spikard-http/Cargo.toml +12 -16
  43. data/vendor/crates/spikard-http/examples/sse-notifications.rs +148 -0
  44. data/vendor/crates/spikard-http/examples/websocket-chat.rs +92 -0
  45. data/vendor/crates/spikard-http/src/auth.rs +65 -16
  46. data/vendor/crates/spikard-http/src/background.rs +1614 -3
  47. data/vendor/crates/spikard-http/src/cors.rs +515 -0
  48. data/vendor/crates/spikard-http/src/debug.rs +65 -0
  49. data/vendor/crates/spikard-http/src/di_handler.rs +1322 -77
  50. data/vendor/crates/spikard-http/src/handler_response.rs +711 -0
  51. data/vendor/crates/spikard-http/src/handler_trait.rs +607 -5
  52. data/vendor/crates/spikard-http/src/handler_trait_tests.rs +6 -0
  53. data/vendor/crates/spikard-http/src/lib.rs +33 -28
  54. data/vendor/crates/spikard-http/src/lifecycle/adapter.rs +81 -0
  55. data/vendor/crates/spikard-http/src/lifecycle.rs +765 -0
  56. data/vendor/crates/spikard-http/src/middleware/mod.rs +372 -117
  57. data/vendor/crates/spikard-http/src/middleware/multipart.rs +836 -10
  58. data/vendor/crates/spikard-http/src/middleware/urlencoded.rs +409 -43
  59. data/vendor/crates/spikard-http/src/middleware/validation.rs +513 -65
  60. data/vendor/crates/spikard-http/src/openapi/parameter_extraction.rs +345 -0
  61. data/vendor/crates/spikard-http/src/openapi/schema_conversion.rs +1055 -0
  62. data/vendor/crates/spikard-http/src/openapi/spec_generation.rs +473 -3
  63. data/vendor/crates/spikard-http/src/query_parser.rs +455 -31
  64. data/vendor/crates/spikard-http/src/response.rs +321 -0
  65. data/vendor/crates/spikard-http/src/server/handler.rs +1572 -9
  66. data/vendor/crates/spikard-http/src/server/lifecycle_execution.rs +136 -0
  67. data/vendor/crates/spikard-http/src/server/mod.rs +875 -178
  68. data/vendor/crates/spikard-http/src/server/request_extraction.rs +674 -23
  69. data/vendor/crates/spikard-http/src/server/routing_factory.rs +599 -0
  70. data/vendor/crates/spikard-http/src/sse.rs +983 -21
  71. data/vendor/crates/spikard-http/src/testing/form.rs +38 -0
  72. data/vendor/crates/spikard-http/src/testing/test_client.rs +0 -2
  73. data/vendor/crates/spikard-http/src/testing.rs +7 -7
  74. data/vendor/crates/spikard-http/src/websocket.rs +1055 -4
  75. data/vendor/crates/spikard-http/tests/background_behavior.rs +832 -0
  76. data/vendor/crates/spikard-http/tests/common/handlers.rs +309 -0
  77. data/vendor/crates/spikard-http/tests/common/mod.rs +26 -0
  78. data/vendor/crates/spikard-http/tests/di_integration.rs +192 -0
  79. data/vendor/crates/spikard-http/tests/doc_snippets.rs +5 -0
  80. data/vendor/crates/spikard-http/tests/lifecycle_execution.rs +1093 -0
  81. data/vendor/crates/spikard-http/tests/multipart_behavior.rs +656 -0
  82. data/vendor/crates/spikard-http/tests/server_config_builder.rs +314 -0
  83. data/vendor/crates/spikard-http/tests/sse_behavior.rs +620 -0
  84. data/vendor/crates/spikard-http/tests/websocket_behavior.rs +663 -0
  85. data/vendor/crates/spikard-rb/Cargo.toml +10 -4
  86. data/vendor/crates/spikard-rb/build.rs +196 -5
  87. data/vendor/crates/spikard-rb/src/config/mod.rs +5 -0
  88. data/vendor/crates/spikard-rb/src/{config.rs → config/server_config.rs} +100 -109
  89. data/vendor/crates/spikard-rb/src/conversion.rs +121 -20
  90. data/vendor/crates/spikard-rb/src/di/builder.rs +100 -0
  91. data/vendor/crates/spikard-rb/src/{di.rs → di/mod.rs} +12 -46
  92. data/vendor/crates/spikard-rb/src/handler.rs +100 -107
  93. data/vendor/crates/spikard-rb/src/integration/mod.rs +3 -0
  94. data/vendor/crates/spikard-rb/src/lib.rs +467 -1428
  95. data/vendor/crates/spikard-rb/src/lifecycle.rs +1 -0
  96. data/vendor/crates/spikard-rb/src/metadata/mod.rs +5 -0
  97. data/vendor/crates/spikard-rb/src/metadata/route_extraction.rs +447 -0
  98. data/vendor/crates/spikard-rb/src/runtime/mod.rs +5 -0
  99. data/vendor/crates/spikard-rb/src/runtime/server_runner.rs +324 -0
  100. data/vendor/crates/spikard-rb/src/server.rs +47 -22
  101. data/vendor/crates/spikard-rb/src/{test_client.rs → testing/client.rs} +187 -40
  102. data/vendor/crates/spikard-rb/src/testing/mod.rs +7 -0
  103. data/vendor/crates/spikard-rb/src/testing/websocket.rs +635 -0
  104. data/vendor/crates/spikard-rb/src/websocket.rs +178 -37
  105. metadata +46 -13
  106. data/vendor/crates/spikard-http/src/parameters.rs +0 -1
  107. data/vendor/crates/spikard-http/src/problem.rs +0 -1
  108. data/vendor/crates/spikard-http/src/router.rs +0 -1
  109. data/vendor/crates/spikard-http/src/schema_registry.rs +0 -1
  110. data/vendor/crates/spikard-http/src/type_hints.rs +0 -1
  111. data/vendor/crates/spikard-http/src/validation.rs +0 -1
  112. data/vendor/crates/spikard-rb/src/test_websocket.rs +0 -221
  113. /data/vendor/crates/spikard-rb/src/{test_sse.rs → testing/sse.rs} +0 -0
@@ -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
+ }
@@ -1,6 +1,6 @@
1
1
  [package]
2
2
  name = "spikard-rb"
3
- version = "0.3.6"
3
+ version = "0.5.0"
4
4
  edition = "2024"
5
5
  authors = ["Na'aman Hirschfeld <nhirschfeld@gmail.com>"]
6
6
  license = "MIT"
@@ -11,28 +11,34 @@ keywords = ["ruby", "bindings", "magnus", "http", "framework"]
11
11
  categories = ["api-bindings", "web-programming::http-server"]
12
12
  documentation = "https://docs.rs/spikard-rb"
13
13
  readme = "README.md"
14
+ build = "build.rs"
14
15
 
15
16
  [lib]
16
17
  name = "spikard_rb_core"
17
- crate-type = ["cdylib", "rlib"]
18
18
 
19
19
  [dependencies]
20
- magnus = { git = "https://github.com/matsadler/magnus", rev = "f6db11769efb517427bf7f121f9c32e18b059b38", features = ["rb-sys"] }
20
+ magnus = { version = "0.8.2", features = ["rb-sys", "embed"] }
21
21
  serde = { version = "1.0", features = ["derive"] }
22
22
  serde_json = "1.0"
23
23
  spikard-core = { path = "../spikard-core", features = ["di"] }
24
24
  spikard-http = { path = "../spikard-http", features = ["di"] }
25
- axum = "0.8"
25
+ spikard-bindings-shared = { path = "../spikard-bindings-shared" }
26
+ spikard-rb-macros = { path = "../spikard-rb-macros" }
27
+ axum = { version = "0.8", features = ["multipart", "ws"] }
26
28
  axum-test = "18"
27
29
  bytes = "1.11"
28
30
  cookie = "0.18"
29
31
  tokio = { version = "1", features = ["full"] }
32
+ tungstenite = "0.24"
30
33
  tracing = "0.1"
31
34
  serde_qs = "0.15"
32
35
  urlencoding = "2.1"
33
36
  once_cell = "1.21"
34
37
  async-stream = "0.3"
35
38
  http = "1.4"
39
+ paste = "1.0.15"
40
+ rb-sys = "0"
41
+ url = "2.5"
36
42
 
37
43
  [features]
38
44
  default = ["di"]