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,221 +1,256 @@
1
- # frozen_string_literal: true
2
-
3
- require 'json'
4
-
5
- module Spikard
6
- # Testing helpers that wrap the native Ruby extension.
7
- module Testing
8
- module_function
9
-
10
- def create_test_client(app, config: nil)
11
- unless defined?(Spikard::Native::TestClient)
12
- raise LoadError, 'Spikard native test client is not available. Build the native extension before running tests.'
13
- end
14
-
15
- # Allow generated apps to stash a test config
16
- if config.nil? && app.instance_variable_defined?(:@__spikard_test_config)
17
- config = app.instance_variable_get(:@__spikard_test_config)
18
- end
19
-
20
- # Use default config if none provided
21
- config ||= Spikard::ServerConfig.new
22
-
23
- routes_json = app.normalized_routes_json
24
- handlers = app.handler_map.transform_keys(&:to_sym)
25
- ws_handlers = app.websocket_handlers || {}
26
- sse_producers = app.sse_producers || {}
27
- dependencies = app.dependencies || {}
28
- native = Spikard::Native::TestClient.new(routes_json, handlers, config, ws_handlers, sse_producers, dependencies)
29
- TestClient.new(native)
30
- end
31
-
32
- # High level wrapper around the native test client.
33
- class TestClient
34
- def initialize(native)
35
- @native = native
36
- end
37
-
38
- # Factory method for creating test client from an app
39
- def self.new(app_or_native, config: nil)
40
- # If passed a native client directly, use it
41
- return super(app_or_native) if app_or_native.is_a?(Spikard::Native::TestClient)
42
-
43
- # Otherwise, create test client from app
44
- Spikard::Testing.create_test_client(app_or_native, config: config)
45
- end
46
-
47
- def request(method, path, **options)
48
- payload = @native.request(method.to_s.upcase, path, options)
49
- Response.new(payload)
50
- end
51
-
52
- def websocket(path)
53
- native_ws = @native.websocket(path)
54
- WebSocketTestConnection.new(native_ws)
55
- end
56
-
57
- def sse(path)
58
- native_sse = @native.sse(path)
59
- SseStream.new(native_sse)
60
- end
61
-
62
- def close
63
- @native.close
64
- end
65
-
66
- %w[get post put patch delete head options trace].each do |verb|
67
- define_method(verb) do |path, **options|
68
- request(verb.upcase, path, **options)
69
- end
70
- end
71
- end
72
-
73
- # WebSocket test connection wrapper
74
- class WebSocketTestConnection
75
- def initialize(native_ws)
76
- @native_ws = native_ws
77
- end
78
-
79
- def send_text(text)
80
- @native_ws.send_text(JSON.generate(text))
81
- end
82
-
83
- def send_json(obj)
84
- @native_ws.send_json(obj)
85
- end
86
-
87
- def receive_text
88
- raw = @native_ws.receive_text
89
- JSON.parse(raw)
90
- rescue JSON::ParserError
91
- raw
92
- end
93
-
94
- def receive_json
95
- @native_ws.receive_json
96
- end
97
-
98
- def receive_bytes
99
- receive_text
100
- end
101
-
102
- def receive_message
103
- native_msg = @native_ws.receive_message
104
- WebSocketMessage.new(native_msg)
105
- end
106
-
107
- def close
108
- @native_ws.close
109
- end
110
- end
111
-
112
- # WebSocket message wrapper
113
- class WebSocketMessage
114
- def initialize(native_msg)
115
- @native_msg = native_msg
116
- end
117
-
118
- def as_text
119
- raw = @native_msg.as_text
120
- return unless raw
121
-
122
- JSON.parse(raw)
123
- rescue JSON::ParserError
124
- raw
125
- end
126
-
127
- def as_json
128
- @native_msg.as_json
129
- end
130
-
131
- def as_binary
132
- @native_msg.as_binary
133
- end
134
-
135
- def close?
136
- @native_msg.is_close
137
- end
138
- end
139
-
140
- # SSE stream wrapper
141
- class SseStream
142
- def initialize(native_sse)
143
- @native_sse = native_sse
144
- end
145
-
146
- def body
147
- @native_sse.body
148
- end
149
-
150
- def events
151
- parsed_chunks.map { |chunk| InlineSseEvent.new(chunk) }
152
- end
153
-
154
- def events_as_json
155
- parsed_chunks.filter_map do |chunk|
156
- JSON.parse(chunk)
157
- rescue JSON::ParserError
158
- nil
159
- end
160
- end
161
-
162
- private
163
-
164
- # rubocop:disable Metrics/AbcSize, Metrics/MethodLength
165
- def parsed_chunks
166
- raw = body.to_s.gsub("\r\n", "\n")
167
- events = []
168
- current = []
169
-
170
- raw.each_line do |line|
171
- stripped = line.chomp
172
- if stripped.start_with?('data:')
173
- current << stripped[5..].strip
174
- elsif stripped.empty?
175
- unless current.empty?
176
- data = current.join("\n").strip
177
- events << data unless data.empty?
178
- current = []
179
- end
180
- end
181
- end
182
-
183
- unless current.empty?
184
- data = current.join("\n").strip
185
- events << data unless data.empty?
186
- end
187
-
188
- events
189
- end
190
- # rubocop:enable Metrics/AbcSize, Metrics/MethodLength
191
- end
192
-
193
- # SSE event wrapper
194
- class SseEvent
195
- def initialize(native_event)
196
- @native_event = native_event
197
- end
198
-
199
- def data
200
- @native_event.data
201
- end
202
-
203
- def as_json
204
- @native_event.as_json
205
- end
206
- end
207
-
208
- # Lightweight wrapper for parsed SSE events backed by strings.
209
- class InlineSseEvent
210
- attr_reader :data
211
-
212
- def initialize(data)
213
- @data = data
214
- end
215
-
216
- def as_json
217
- JSON.parse(@data)
218
- end
219
- end
220
- end
221
- end
1
+ # frozen_string_literal: true
2
+
3
+ require 'json'
4
+ require 'timeout'
5
+
6
+ module Spikard
7
+ # Testing helpers that wrap the native Ruby extension.
8
+ module Testing
9
+ module_function
10
+
11
+ def create_test_client(app, config: nil)
12
+ trace('create_test_client:start')
13
+ ensure_native_test_client!
14
+ config = resolve_test_config(app, config)
15
+ native = build_native_test_client(app, config)
16
+ trace('create_test_client:done')
17
+ TestClient.new(native)
18
+ end
19
+
20
+ def ensure_native_test_client!
21
+ return if defined?(Spikard::Native::TestClient)
22
+
23
+ raise LoadError, 'Spikard native test client is not available. Build the native extension before running tests.'
24
+ end
25
+
26
+ def resolve_test_config(app, config)
27
+ return config if config
28
+
29
+ if app.instance_variable_defined?(:@__spikard_test_config)
30
+ return app.instance_variable_get(:@__spikard_test_config)
31
+ end
32
+
33
+ Spikard::ServerConfig.new
34
+ end
35
+
36
+ def build_native_test_client(app, config)
37
+ routes_json = app.normalized_routes_json
38
+ handlers = app.handler_map.transform_keys(&:to_sym)
39
+ ws_handlers = app.websocket_handlers || {}
40
+ sse_producers = app.sse_producers || {}
41
+ dependencies = app.dependencies || {}
42
+ Spikard::Native::TestClient.new(routes_json, handlers, config, ws_handlers, sse_producers, dependencies)
43
+ end
44
+
45
+ # High level wrapper around the native test client.
46
+ class TestClient
47
+ def initialize(native)
48
+ @native = native
49
+ end
50
+
51
+ # Factory method for creating test client from an app
52
+ def self.new(app_or_native, config: nil)
53
+ # If passed a native client directly, use it
54
+ return super(app_or_native) if app_or_native.is_a?(Spikard::Native::TestClient)
55
+
56
+ # Otherwise, create test client from app
57
+ Spikard::Testing.create_test_client(app_or_native, config: config)
58
+ end
59
+
60
+ def request(method, path, **options)
61
+ payload = @native.request(method.to_s.upcase, path, options)
62
+ Response.new(payload)
63
+ end
64
+
65
+ def websocket(path)
66
+ Testing.trace("websocket:start #{path}")
67
+ native_ws = @native.websocket(path)
68
+ Testing.trace("websocket:connected #{path}")
69
+ WebSocketTestConnection.new(native_ws)
70
+ end
71
+
72
+ def sse(path)
73
+ native_sse = @native.sse(path)
74
+ SseStream.new(native_sse)
75
+ end
76
+
77
+ def close
78
+ @native.close
79
+ end
80
+
81
+ %w[get post put patch delete head options trace].each do |verb|
82
+ define_method(verb) do |path, **options|
83
+ request(verb.upcase, path, **options)
84
+ end
85
+ end
86
+ end
87
+
88
+ # WebSocket test connection wrapper
89
+ class WebSocketTestConnection
90
+ def initialize(native_ws)
91
+ @native_ws = native_ws
92
+ end
93
+
94
+ def send_text(text)
95
+ Testing.trace('websocket:send_text')
96
+ @native_ws.send_text(JSON.generate(text))
97
+ end
98
+
99
+ def send_json(obj)
100
+ Testing.trace('websocket:send_json')
101
+ @native_ws.send_json(obj)
102
+ end
103
+
104
+ def receive_text
105
+ Testing.trace('websocket:receive_text')
106
+ raw = with_timeout { @native_ws.receive_text }
107
+ JSON.parse(raw)
108
+ rescue JSON::ParserError
109
+ raw
110
+ end
111
+
112
+ def receive_json
113
+ Testing.trace('websocket:receive_json')
114
+ with_timeout { @native_ws.receive_json }
115
+ end
116
+
117
+ def receive_bytes
118
+ receive_text
119
+ end
120
+
121
+ def receive_message
122
+ native_msg = @native_ws.receive_message
123
+ WebSocketMessage.new(native_msg)
124
+ end
125
+
126
+ def close
127
+ Testing.trace('websocket:close')
128
+ @native_ws.close
129
+ end
130
+
131
+ private
132
+
133
+ def with_timeout(&)
134
+ timeout_ms = ENV.fetch('SPIKARD_RB_TEST_TIMEOUT_MS', nil)
135
+ return yield unless timeout_ms
136
+
137
+ Timeout.timeout(timeout_ms.to_f / 1000.0, &)
138
+ end
139
+ end
140
+
141
+ # WebSocket message wrapper
142
+ class WebSocketMessage
143
+ def initialize(native_msg)
144
+ @native_msg = native_msg
145
+ end
146
+
147
+ def as_text
148
+ raw = @native_msg.as_text
149
+ return unless raw
150
+
151
+ JSON.parse(raw)
152
+ rescue JSON::ParserError
153
+ raw
154
+ end
155
+
156
+ def as_json
157
+ @native_msg.as_json
158
+ end
159
+
160
+ def as_binary
161
+ @native_msg.as_binary
162
+ end
163
+
164
+ def close?
165
+ @native_msg.is_close
166
+ end
167
+ end
168
+
169
+ # SSE stream wrapper
170
+ class SseStream
171
+ def initialize(native_sse)
172
+ @native_sse = native_sse
173
+ end
174
+
175
+ def body
176
+ @native_sse.body
177
+ end
178
+
179
+ def events
180
+ parsed_chunks.map { |chunk| InlineSseEvent.new(chunk) }
181
+ end
182
+
183
+ def events_as_json
184
+ parsed_chunks.filter_map do |chunk|
185
+ JSON.parse(chunk)
186
+ rescue JSON::ParserError
187
+ nil
188
+ end
189
+ end
190
+
191
+ private
192
+
193
+ # rubocop:disable Metrics/AbcSize, Metrics/MethodLength
194
+ def parsed_chunks
195
+ raw = body.to_s.gsub("\r\n", "\n")
196
+ events = []
197
+ current = []
198
+
199
+ raw.each_line do |line|
200
+ stripped = line.chomp
201
+ if stripped.start_with?('data:')
202
+ current << stripped[5..].strip
203
+ elsif stripped.empty?
204
+ unless current.empty?
205
+ data = current.join("\n").strip
206
+ events << data unless data.empty?
207
+ current = []
208
+ end
209
+ end
210
+ end
211
+
212
+ unless current.empty?
213
+ data = current.join("\n").strip
214
+ events << data unless data.empty?
215
+ end
216
+
217
+ events
218
+ end
219
+ # rubocop:enable Metrics/AbcSize, Metrics/MethodLength
220
+ end
221
+
222
+ # SSE event wrapper
223
+ class SseEvent
224
+ def initialize(native_event)
225
+ @native_event = native_event
226
+ end
227
+
228
+ def data
229
+ @native_event.data
230
+ end
231
+
232
+ def as_json
233
+ @native_event.as_json
234
+ end
235
+ end
236
+
237
+ # Lightweight wrapper for parsed SSE events backed by strings.
238
+ class InlineSseEvent
239
+ attr_reader :data
240
+
241
+ def initialize(data)
242
+ @data = data
243
+ end
244
+
245
+ def as_json
246
+ JSON.parse(@data)
247
+ end
248
+ end
249
+
250
+ def trace(message)
251
+ return unless ENV['SPIKARD_RB_TEST_TRACE'] == '1'
252
+
253
+ warn("[spikard-rb-test] #{message}")
254
+ end
255
+ end
256
+ end