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
@@ -1,231 +1,231 @@
1
- //! Ruby SSE producer bindings
2
- //!
3
- //! This module provides the bridge between Ruby blocks/procs and Rust's SSE system.
4
- //! Uses magnus to safely call Ruby code from Rust async tasks.
5
-
6
- use magnus::{RHash, Value, prelude::*, value::Opaque};
7
- use serde_json::Value as JsonValue;
8
- use spikard_http::{SseEvent, SseEventProducer};
9
- use tracing::{debug, error};
10
-
11
- /// Ruby implementation of SseEventProducer
12
- pub struct RubySseEventProducer {
13
- /// Producer name for debugging
14
- name: String,
15
- /// Ruby proc/callable for next_event (Opaque for Send safety)
16
- next_event_proc: Opaque<Value>,
17
- /// Ruby proc/callable for on_connect (Opaque for Send safety)
18
- on_connect_proc: Option<Opaque<Value>>,
19
- /// Ruby proc/callable for on_disconnect (Opaque for Send safety)
20
- on_disconnect_proc: Option<Opaque<Value>>,
21
- }
22
-
23
- impl RubySseEventProducer {
24
- /// Create a new Ruby SSE event producer
25
- #[allow(dead_code)]
26
- pub fn new(
27
- name: String,
28
- next_event_proc: Value,
29
- on_connect_proc: Option<Value>,
30
- on_disconnect_proc: Option<Value>,
31
- ) -> Self {
32
- Self {
33
- name,
34
- next_event_proc: next_event_proc.into(),
35
- on_connect_proc: on_connect_proc.map(|v| v.into()),
36
- on_disconnect_proc: on_disconnect_proc.map(|v| v.into()),
37
- }
38
- }
39
-
40
- /// Convert Ruby value to JSON
41
- fn ruby_to_json(ruby: &magnus::Ruby, value: Value) -> Result<JsonValue, String> {
42
- if value.is_nil() {
43
- return Ok(JsonValue::Null);
44
- }
45
-
46
- let json_module: Value = ruby
47
- .class_object()
48
- .const_get("JSON")
49
- .map_err(|e| format!("JSON module not available: {}", e))?;
50
-
51
- let json_str: String = json_module
52
- .funcall("generate", (value,))
53
- .map_err(|e| format!("Failed to generate JSON: {}", e))?;
54
-
55
- serde_json::from_str(&json_str).map_err(|e| format!("Failed to parse JSON: {}", e))
56
- }
57
- }
58
-
59
- impl SseEventProducer for RubySseEventProducer {
60
- async fn next_event(&self) -> Option<SseEvent> {
61
- debug!("Ruby SSE producer '{}': next_event", self.name);
62
-
63
- match magnus::Ruby::get()
64
- .map_err(|e| format!("Failed to get Ruby: {}", e))
65
- .and_then(|ruby| {
66
- debug!("Ruby SSE producer: acquired Ruby VM");
67
-
68
- let proc_value = ruby.get_inner(self.next_event_proc);
69
- let result: Value = proc_value
70
- .funcall("call", ())
71
- .map_err(|e| format!("Producer '{}' call failed: {}", self.name, e))?;
72
-
73
- debug!("Ruby SSE producer: called next_event proc");
74
-
75
- if result.is_nil() {
76
- debug!("Ruby SSE producer: received nil, ending stream");
77
- return Ok(None);
78
- }
79
-
80
- let result_hash = if let Some(hash) = RHash::from_value(result) {
81
- hash
82
- } else {
83
- let hash_value: Value = result.funcall("to_h", ()).map_err(|e| {
84
- format!(
85
- "next_event must return a Hash/SseEvent convertible via to_h ({}): {}",
86
- unsafe { result.classname() },
87
- e
88
- )
89
- })?;
90
- RHash::from_value(hash_value).ok_or_else(|| {
91
- format!("next_event to_h must return a Hash, got {}", unsafe {
92
- hash_value.classname()
93
- })
94
- })?
95
- };
96
-
97
- let data_value = result_hash
98
- .get(ruby.to_symbol("data"))
99
- .ok_or_else(|| "next_event Hash must have :data key".to_string())?;
100
-
101
- let data_json = Self::ruby_to_json(&ruby, data_value)?;
102
-
103
- let event_type: Option<String> = result_hash
104
- .get(ruby.to_symbol("event_type"))
105
- .and_then(|v| if v.is_nil() { None } else { String::try_convert(v).ok() });
106
-
107
- let id: Option<String> = result_hash
108
- .get(ruby.to_symbol("id"))
109
- .and_then(|v| if v.is_nil() { None } else { String::try_convert(v).ok() });
110
-
111
- let retry: Option<u64> = result_hash
112
- .get(ruby.to_symbol("retry"))
113
- .and_then(|v| if v.is_nil() { None } else { u64::try_convert(v).ok() });
114
-
115
- let mut event = if let Some(et) = event_type {
116
- SseEvent::with_type(et, data_json)
117
- } else {
118
- SseEvent::new(data_json)
119
- };
120
-
121
- if let Some(id_str) = id {
122
- event = event.with_id(id_str);
123
- }
124
-
125
- if let Some(retry_ms) = retry {
126
- event = event.with_retry(retry_ms);
127
- }
128
-
129
- Ok(Some(event))
130
- }) {
131
- Ok(event) => event,
132
- Err(e) => {
133
- error!("Ruby error in next_event: {}", e);
134
- None
135
- }
136
- }
137
- }
138
-
139
- async fn on_connect(&self) {
140
- debug!("Ruby SSE producer '{}': on_connect", self.name);
141
-
142
- if let Some(on_connect_proc) = self.on_connect_proc
143
- && let Err(e) = magnus::Ruby::get()
144
- .map_err(|e| format!("Failed to get Ruby: {}", e))
145
- .and_then(|ruby| {
146
- debug!("Ruby SSE producer: on_connect acquired Ruby VM");
147
- let proc_value = ruby.get_inner(on_connect_proc);
148
- proc_value
149
- .funcall::<_, _, Value>("call", ())
150
- .map_err(|e| format!("on_connect '{}' call failed: {}", self.name, e))?;
151
- debug!("Ruby SSE producer: on_connect completed");
152
- Ok(())
153
- })
154
- {
155
- error!("on_connect error: {}", e);
156
- }
157
- }
158
-
159
- async fn on_disconnect(&self) {
160
- debug!("Ruby SSE producer '{}': on_disconnect", self.name);
161
-
162
- if let Some(on_disconnect_proc) = self.on_disconnect_proc
163
- && let Err(e) = magnus::Ruby::get()
164
- .map_err(|e| format!("Failed to get Ruby: {}", e))
165
- .and_then(|ruby| {
166
- let proc_value = ruby.get_inner(on_disconnect_proc);
167
- proc_value
168
- .funcall::<_, _, Value>("call", ())
169
- .map_err(|e| format!("on_disconnect '{}' call failed: {}", self.name, e))?;
170
- debug!("Ruby SSE producer: on_disconnect completed");
171
- Ok(())
172
- })
173
- {
174
- error!("on_disconnect error: {}", e);
175
- }
176
- }
177
- }
178
-
179
- unsafe impl Send for RubySseEventProducer {}
180
- unsafe impl Sync for RubySseEventProducer {}
181
-
182
- /// Create SseState from Ruby producer object
183
- ///
184
- /// This function is designed to be called from Ruby to register SSE producers.
185
- #[allow(dead_code)]
186
- pub fn create_sse_state(
187
- ruby: &magnus::Ruby,
188
- producer_obj: Value,
189
- ) -> Result<spikard_http::SseState<RubySseEventProducer>, magnus::Error> {
190
- let next_event_proc: Value = producer_obj
191
- .funcall("method", (ruby.to_symbol("next_event"),))
192
- .map_err(|e| {
193
- magnus::Error::new(
194
- ruby.exception_arg_error(),
195
- format!("next_event method not found: {}", e),
196
- )
197
- })?;
198
-
199
- let on_connect_proc = producer_obj
200
- .funcall::<_, _, Value>("method", (ruby.to_symbol("on_connect"),))
201
- .ok();
202
-
203
- let on_disconnect_proc = producer_obj
204
- .funcall::<_, _, Value>("method", (ruby.to_symbol("on_disconnect"),))
205
- .ok();
206
-
207
- let event_schema = producer_obj
208
- .funcall::<_, _, Value>("instance_variable_get", (ruby.to_symbol("@_event_schema"),))
209
- .ok()
210
- .and_then(|v| {
211
- if v.is_nil() {
212
- None
213
- } else {
214
- RubySseEventProducer::ruby_to_json(ruby, v).ok()
215
- }
216
- });
217
-
218
- let ruby_producer = RubySseEventProducer::new(
219
- "SseEventProducer".to_string(),
220
- next_event_proc,
221
- on_connect_proc,
222
- on_disconnect_proc,
223
- );
224
-
225
- if event_schema.is_some() {
226
- spikard_http::SseState::with_schema(ruby_producer, event_schema)
227
- .map_err(|e| magnus::Error::new(ruby.exception_runtime_error(), e))
228
- } else {
229
- Ok(spikard_http::SseState::new(ruby_producer))
230
- }
231
- }
1
+ //! Ruby SSE producer bindings
2
+ //!
3
+ //! This module provides the bridge between Ruby blocks/procs and Rust's SSE system.
4
+ //! Uses magnus to safely call Ruby code from Rust async tasks.
5
+
6
+ use magnus::{RHash, Value, prelude::*, value::Opaque};
7
+ use serde_json::Value as JsonValue;
8
+ use spikard_http::{SseEvent, SseEventProducer};
9
+ use tracing::{debug, error};
10
+
11
+ /// Ruby implementation of SseEventProducer
12
+ pub struct RubySseEventProducer {
13
+ /// Producer name for debugging
14
+ name: String,
15
+ /// Ruby proc/callable for next_event (Opaque for Send safety)
16
+ next_event_proc: Opaque<Value>,
17
+ /// Ruby proc/callable for on_connect (Opaque for Send safety)
18
+ on_connect_proc: Option<Opaque<Value>>,
19
+ /// Ruby proc/callable for on_disconnect (Opaque for Send safety)
20
+ on_disconnect_proc: Option<Opaque<Value>>,
21
+ }
22
+
23
+ impl RubySseEventProducer {
24
+ /// Create a new Ruby SSE event producer
25
+ #[allow(dead_code)]
26
+ pub fn new(
27
+ name: String,
28
+ next_event_proc: Value,
29
+ on_connect_proc: Option<Value>,
30
+ on_disconnect_proc: Option<Value>,
31
+ ) -> Self {
32
+ Self {
33
+ name,
34
+ next_event_proc: next_event_proc.into(),
35
+ on_connect_proc: on_connect_proc.map(|v| v.into()),
36
+ on_disconnect_proc: on_disconnect_proc.map(|v| v.into()),
37
+ }
38
+ }
39
+
40
+ /// Convert Ruby value to JSON
41
+ fn ruby_to_json(ruby: &magnus::Ruby, value: Value) -> Result<JsonValue, String> {
42
+ if value.is_nil() {
43
+ return Ok(JsonValue::Null);
44
+ }
45
+
46
+ let json_module: Value = ruby
47
+ .class_object()
48
+ .const_get("JSON")
49
+ .map_err(|e| format!("JSON module not available: {}", e))?;
50
+
51
+ let json_str: String = json_module
52
+ .funcall("generate", (value,))
53
+ .map_err(|e| format!("Failed to generate JSON: {}", e))?;
54
+
55
+ serde_json::from_str(&json_str).map_err(|e| format!("Failed to parse JSON: {}", e))
56
+ }
57
+ }
58
+
59
+ impl SseEventProducer for RubySseEventProducer {
60
+ async fn next_event(&self) -> Option<SseEvent> {
61
+ debug!("Ruby SSE producer '{}': next_event", self.name);
62
+
63
+ match magnus::Ruby::get()
64
+ .map_err(|e| format!("Failed to get Ruby: {}", e))
65
+ .and_then(|ruby| {
66
+ debug!("Ruby SSE producer: acquired Ruby VM");
67
+
68
+ let proc_value = ruby.get_inner(self.next_event_proc);
69
+ let result: Value = proc_value
70
+ .funcall("call", ())
71
+ .map_err(|e| format!("Producer '{}' call failed: {}", self.name, e))?;
72
+
73
+ debug!("Ruby SSE producer: called next_event proc");
74
+
75
+ if result.is_nil() {
76
+ debug!("Ruby SSE producer: received nil, ending stream");
77
+ return Ok(None);
78
+ }
79
+
80
+ let result_hash = if let Some(hash) = RHash::from_value(result) {
81
+ hash
82
+ } else {
83
+ let hash_value: Value = result.funcall("to_h", ()).map_err(|e| {
84
+ format!(
85
+ "next_event must return a Hash/SseEvent convertible via to_h ({}): {}",
86
+ unsafe { result.classname() },
87
+ e
88
+ )
89
+ })?;
90
+ RHash::from_value(hash_value).ok_or_else(|| {
91
+ format!("next_event to_h must return a Hash, got {}", unsafe {
92
+ hash_value.classname()
93
+ })
94
+ })?
95
+ };
96
+
97
+ let data_value = result_hash
98
+ .get(ruby.to_symbol("data"))
99
+ .ok_or_else(|| "next_event Hash must have :data key".to_string())?;
100
+
101
+ let data_json = Self::ruby_to_json(&ruby, data_value)?;
102
+
103
+ let event_type: Option<String> = result_hash
104
+ .get(ruby.to_symbol("event_type"))
105
+ .and_then(|v| if v.is_nil() { None } else { String::try_convert(v).ok() });
106
+
107
+ let id: Option<String> = result_hash
108
+ .get(ruby.to_symbol("id"))
109
+ .and_then(|v| if v.is_nil() { None } else { String::try_convert(v).ok() });
110
+
111
+ let retry: Option<u64> = result_hash
112
+ .get(ruby.to_symbol("retry"))
113
+ .and_then(|v| if v.is_nil() { None } else { u64::try_convert(v).ok() });
114
+
115
+ let mut event = if let Some(et) = event_type {
116
+ SseEvent::with_type(et, data_json)
117
+ } else {
118
+ SseEvent::new(data_json)
119
+ };
120
+
121
+ if let Some(id_str) = id {
122
+ event = event.with_id(id_str);
123
+ }
124
+
125
+ if let Some(retry_ms) = retry {
126
+ event = event.with_retry(retry_ms);
127
+ }
128
+
129
+ Ok(Some(event))
130
+ }) {
131
+ Ok(event) => event,
132
+ Err(e) => {
133
+ error!("Ruby error in next_event: {}", e);
134
+ None
135
+ }
136
+ }
137
+ }
138
+
139
+ async fn on_connect(&self) {
140
+ debug!("Ruby SSE producer '{}': on_connect", self.name);
141
+
142
+ if let Some(on_connect_proc) = self.on_connect_proc
143
+ && let Err(e) = magnus::Ruby::get()
144
+ .map_err(|e| format!("Failed to get Ruby: {}", e))
145
+ .and_then(|ruby| {
146
+ debug!("Ruby SSE producer: on_connect acquired Ruby VM");
147
+ let proc_value = ruby.get_inner(on_connect_proc);
148
+ proc_value
149
+ .funcall::<_, _, Value>("call", ())
150
+ .map_err(|e| format!("on_connect '{}' call failed: {}", self.name, e))?;
151
+ debug!("Ruby SSE producer: on_connect completed");
152
+ Ok(())
153
+ })
154
+ {
155
+ error!("on_connect error: {}", e);
156
+ }
157
+ }
158
+
159
+ async fn on_disconnect(&self) {
160
+ debug!("Ruby SSE producer '{}': on_disconnect", self.name);
161
+
162
+ if let Some(on_disconnect_proc) = self.on_disconnect_proc
163
+ && let Err(e) = magnus::Ruby::get()
164
+ .map_err(|e| format!("Failed to get Ruby: {}", e))
165
+ .and_then(|ruby| {
166
+ let proc_value = ruby.get_inner(on_disconnect_proc);
167
+ proc_value
168
+ .funcall::<_, _, Value>("call", ())
169
+ .map_err(|e| format!("on_disconnect '{}' call failed: {}", self.name, e))?;
170
+ debug!("Ruby SSE producer: on_disconnect completed");
171
+ Ok(())
172
+ })
173
+ {
174
+ error!("on_disconnect error: {}", e);
175
+ }
176
+ }
177
+ }
178
+
179
+ unsafe impl Send for RubySseEventProducer {}
180
+ unsafe impl Sync for RubySseEventProducer {}
181
+
182
+ /// Create SseState from Ruby producer object
183
+ ///
184
+ /// This function is designed to be called from Ruby to register SSE producers.
185
+ #[allow(dead_code)]
186
+ pub fn create_sse_state(
187
+ ruby: &magnus::Ruby,
188
+ producer_obj: Value,
189
+ ) -> Result<spikard_http::SseState<RubySseEventProducer>, magnus::Error> {
190
+ let next_event_proc: Value = producer_obj
191
+ .funcall("method", (ruby.to_symbol("next_event"),))
192
+ .map_err(|e| {
193
+ magnus::Error::new(
194
+ ruby.exception_arg_error(),
195
+ format!("next_event method not found: {}", e),
196
+ )
197
+ })?;
198
+
199
+ let on_connect_proc = producer_obj
200
+ .funcall::<_, _, Value>("method", (ruby.to_symbol("on_connect"),))
201
+ .ok();
202
+
203
+ let on_disconnect_proc = producer_obj
204
+ .funcall::<_, _, Value>("method", (ruby.to_symbol("on_disconnect"),))
205
+ .ok();
206
+
207
+ let event_schema = producer_obj
208
+ .funcall::<_, _, Value>("instance_variable_get", (ruby.to_symbol("@_event_schema"),))
209
+ .ok()
210
+ .and_then(|v| {
211
+ if v.is_nil() {
212
+ None
213
+ } else {
214
+ RubySseEventProducer::ruby_to_json(ruby, v).ok()
215
+ }
216
+ });
217
+
218
+ let ruby_producer = RubySseEventProducer::new(
219
+ "SseEventProducer".to_string(),
220
+ next_event_proc,
221
+ on_connect_proc,
222
+ on_disconnect_proc,
223
+ );
224
+
225
+ if event_schema.is_some() {
226
+ spikard_http::SseState::with_schema(ruby_producer, event_schema)
227
+ .map_err(|e| magnus::Error::new(ruby.exception_runtime_error(), e))
228
+ } else {
229
+ Ok(spikard_http::SseState::new(ruby_producer))
230
+ }
231
+ }