spikard 0.12.0 → 0.15.2
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.
- checksums.yaml +4 -4
- data/Steepfile +6 -0
- data/ext/spikard_rb/extconf.rb +1 -2
- data/ext/spikard_rb/{Cargo.lock → native/Cargo.lock} +897 -451
- data/ext/spikard_rb/native/Cargo.toml +24 -0
- data/ext/spikard_rb/src/lib.rs +5366 -3
- data/lib/spikard/native.rb +86 -0
- data/lib/spikard/version.rb +6 -1
- data/lib/spikard.rb +8 -45
- data/lib/spikard_rb.so +0 -0
- data/sig/types.rbs +427 -0
- metadata +14 -242
- data/LICENSE +0 -1
- data/README.md +0 -267
- data/ext/spikard_rb/Cargo.toml +0 -17
- data/lib/spikard/app.rb +0 -428
- data/lib/spikard/background.rb +0 -58
- data/lib/spikard/config.rb +0 -506
- data/lib/spikard/converters.rb +0 -13
- data/lib/spikard/grpc.rb +0 -182
- data/lib/spikard/handler_wrapper.rb +0 -113
- data/lib/spikard/provide.rb +0 -214
- data/lib/spikard/response.rb +0 -173
- data/lib/spikard/schema.rb +0 -243
- data/lib/spikard/sse.rb +0 -111
- data/lib/spikard/streaming_response.rb +0 -44
- data/lib/spikard/testing.rb +0 -432
- data/lib/spikard/upload_file.rb +0 -131
- data/lib/spikard/websocket.rb +0 -59
- data/sig/spikard.rbs +0 -719
- data/vendor/crates/spikard-bindings-shared/Cargo.toml +0 -80
- data/vendor/crates/spikard-bindings-shared/examples/config_extraction.rs +0 -132
- data/vendor/crates/spikard-bindings-shared/src/config_extractor.rs +0 -905
- data/vendor/crates/spikard-bindings-shared/src/conversion_traits.rs +0 -210
- data/vendor/crates/spikard-bindings-shared/src/di_traits.rs +0 -252
- data/vendor/crates/spikard-bindings-shared/src/error_response.rs +0 -404
- data/vendor/crates/spikard-bindings-shared/src/grpc_metadata.rs +0 -199
- data/vendor/crates/spikard-bindings-shared/src/handler_base.rs +0 -252
- data/vendor/crates/spikard-bindings-shared/src/json_conversion.rs +0 -829
- data/vendor/crates/spikard-bindings-shared/src/lazy_cache.rs +0 -587
- data/vendor/crates/spikard-bindings-shared/src/lib.rs +0 -33
- data/vendor/crates/spikard-bindings-shared/src/lifecycle_base.rs +0 -298
- data/vendor/crates/spikard-bindings-shared/src/lifecycle_executor.rs +0 -594
- data/vendor/crates/spikard-bindings-shared/src/response_builder.rs +0 -743
- data/vendor/crates/spikard-bindings-shared/src/response_interpreter.rs +0 -944
- data/vendor/crates/spikard-bindings-shared/src/test_client_base.rs +0 -260
- data/vendor/crates/spikard-bindings-shared/src/validation_helpers.rs +0 -369
- data/vendor/crates/spikard-bindings-shared/tests/config_extractor_behavior.rs +0 -192
- data/vendor/crates/spikard-bindings-shared/tests/error_response_edge_cases.rs +0 -383
- data/vendor/crates/spikard-bindings-shared/tests/full_coverage.rs +0 -459
- data/vendor/crates/spikard-bindings-shared/tests/handler_base_integration.rs +0 -280
- data/vendor/crates/spikard-bindings-shared/tests/integration_tests.rs +0 -669
- data/vendor/crates/spikard-core/Cargo.toml +0 -60
- data/vendor/crates/spikard-core/src/bindings/mod.rs +0 -3
- data/vendor/crates/spikard-core/src/bindings/response.rs +0 -130
- data/vendor/crates/spikard-core/src/debug.rs +0 -127
- data/vendor/crates/spikard-core/src/di/container.rs +0 -702
- data/vendor/crates/spikard-core/src/di/dependency.rs +0 -273
- data/vendor/crates/spikard-core/src/di/error.rs +0 -118
- data/vendor/crates/spikard-core/src/di/factory.rs +0 -538
- data/vendor/crates/spikard-core/src/di/graph.rs +0 -507
- data/vendor/crates/spikard-core/src/di/mod.rs +0 -192
- data/vendor/crates/spikard-core/src/di/resolved.rs +0 -428
- data/vendor/crates/spikard-core/src/di/value.rs +0 -282
- data/vendor/crates/spikard-core/src/errors.rs +0 -72
- data/vendor/crates/spikard-core/src/http.rs +0 -492
- data/vendor/crates/spikard-core/src/lib.rs +0 -29
- data/vendor/crates/spikard-core/src/lifecycle.rs +0 -1273
- data/vendor/crates/spikard-core/src/metadata.rs +0 -378
- data/vendor/crates/spikard-core/src/parameters.rs +0 -2546
- data/vendor/crates/spikard-core/src/problem.rs +0 -358
- data/vendor/crates/spikard-core/src/request_data.rs +0 -1146
- data/vendor/crates/spikard-core/src/router.rs +0 -530
- data/vendor/crates/spikard-core/src/schema_registry.rs +0 -197
- data/vendor/crates/spikard-core/src/type_hints.rs +0 -311
- data/vendor/crates/spikard-core/src/validation/error_mapper.rs +0 -710
- data/vendor/crates/spikard-core/src/validation/mod.rs +0 -470
- data/vendor/crates/spikard-core/tests/bindings_response_tests.rs +0 -136
- data/vendor/crates/spikard-core/tests/di_dependency_defaults.rs +0 -37
- data/vendor/crates/spikard-core/tests/error_mapper.rs +0 -761
- data/vendor/crates/spikard-core/tests/parameters_edge_cases.rs +0 -106
- data/vendor/crates/spikard-core/tests/parameters_full.rs +0 -701
- data/vendor/crates/spikard-core/tests/parameters_schema_and_formats.rs +0 -301
- data/vendor/crates/spikard-core/tests/request_data_roundtrip.rs +0 -67
- data/vendor/crates/spikard-core/tests/validation_coverage.rs +0 -250
- data/vendor/crates/spikard-core/tests/validation_error_paths.rs +0 -45
- data/vendor/crates/spikard-http/Cargo.toml +0 -87
- data/vendor/crates/spikard-http/examples/sse-notifications.rs +0 -148
- data/vendor/crates/spikard-http/examples/websocket-chat.rs +0 -92
- data/vendor/crates/spikard-http/src/auth.rs +0 -301
- data/vendor/crates/spikard-http/src/background.rs +0 -1860
- data/vendor/crates/spikard-http/src/bindings/mod.rs +0 -3
- data/vendor/crates/spikard-http/src/bindings/response.rs +0 -1
- data/vendor/crates/spikard-http/src/body_metadata.rs +0 -8
- data/vendor/crates/spikard-http/src/cors.rs +0 -1026
- data/vendor/crates/spikard-http/src/debug.rs +0 -128
- data/vendor/crates/spikard-http/src/di_handler.rs +0 -1672
- data/vendor/crates/spikard-http/src/grpc/framing.rs +0 -469
- data/vendor/crates/spikard-http/src/grpc/handler.rs +0 -1122
- data/vendor/crates/spikard-http/src/grpc/mod.rs +0 -434
- data/vendor/crates/spikard-http/src/grpc/service.rs +0 -622
- data/vendor/crates/spikard-http/src/grpc/streaming.rs +0 -319
- data/vendor/crates/spikard-http/src/handler_response.rs +0 -901
- data/vendor/crates/spikard-http/src/handler_trait.rs +0 -1015
- data/vendor/crates/spikard-http/src/handler_trait_tests.rs +0 -290
- data/vendor/crates/spikard-http/src/jsonrpc/http_handler.rs +0 -502
- data/vendor/crates/spikard-http/src/jsonrpc/method_registry.rs +0 -648
- data/vendor/crates/spikard-http/src/jsonrpc/mod.rs +0 -58
- data/vendor/crates/spikard-http/src/jsonrpc/protocol.rs +0 -1207
- data/vendor/crates/spikard-http/src/jsonrpc/router.rs +0 -2262
- data/vendor/crates/spikard-http/src/lib.rs +0 -548
- data/vendor/crates/spikard-http/src/lifecycle/adapter.rs +0 -230
- data/vendor/crates/spikard-http/src/lifecycle.rs +0 -1193
- data/vendor/crates/spikard-http/src/middleware/mod.rs +0 -560
- data/vendor/crates/spikard-http/src/middleware/multipart.rs +0 -912
- data/vendor/crates/spikard-http/src/middleware/urlencoded.rs +0 -513
- data/vendor/crates/spikard-http/src/middleware/validation.rs +0 -768
- data/vendor/crates/spikard-http/src/openapi/mod.rs +0 -309
- data/vendor/crates/spikard-http/src/openapi/parameter_extraction.rs +0 -535
- data/vendor/crates/spikard-http/src/openapi/schema_conversion.rs +0 -1363
- data/vendor/crates/spikard-http/src/openapi/spec_generation.rs +0 -667
- data/vendor/crates/spikard-http/src/query_parser.rs +0 -793
- data/vendor/crates/spikard-http/src/response.rs +0 -720
- data/vendor/crates/spikard-http/src/server/fast_router.rs +0 -186
- data/vendor/crates/spikard-http/src/server/grpc_routing.rs +0 -858
- data/vendor/crates/spikard-http/src/server/handler.rs +0 -1661
- data/vendor/crates/spikard-http/src/server/lifecycle_execution.rs +0 -253
- data/vendor/crates/spikard-http/src/server/mod.rs +0 -1649
- data/vendor/crates/spikard-http/src/server/request_extraction.rs +0 -871
- data/vendor/crates/spikard-http/src/server/routing_factory.rs +0 -618
- data/vendor/crates/spikard-http/src/sse.rs +0 -1409
- data/vendor/crates/spikard-http/src/testing/form.rs +0 -52
- data/vendor/crates/spikard-http/src/testing/multipart.rs +0 -64
- data/vendor/crates/spikard-http/src/testing/test_client.rs +0 -787
- data/vendor/crates/spikard-http/src/testing.rs +0 -617
- data/vendor/crates/spikard-http/src/websocket.rs +0 -1477
- data/vendor/crates/spikard-http/tests/auth_integration.rs +0 -645
- data/vendor/crates/spikard-http/tests/background_behavior.rs +0 -832
- data/vendor/crates/spikard-http/tests/common/grpc_helpers.rs +0 -1012
- data/vendor/crates/spikard-http/tests/common/handlers.rs +0 -309
- data/vendor/crates/spikard-http/tests/common/mod.rs +0 -33
- data/vendor/crates/spikard-http/tests/common/test_builders.rs +0 -628
- data/vendor/crates/spikard-http/tests/di_handler_error_responses.rs +0 -162
- data/vendor/crates/spikard-http/tests/di_integration.rs +0 -192
- data/vendor/crates/spikard-http/tests/doc_snippets.rs +0 -5
- data/vendor/crates/spikard-http/tests/grpc_bidirectional_streaming.rs +0 -430
- data/vendor/crates/spikard-http/tests/grpc_client_streaming.rs +0 -738
- data/vendor/crates/spikard-http/tests/grpc_error_handling_test.rs +0 -652
- data/vendor/crates/spikard-http/tests/grpc_integration_test.rs +0 -334
- data/vendor/crates/spikard-http/tests/grpc_metadata_test.rs +0 -532
- data/vendor/crates/spikard-http/tests/grpc_server_integration.rs +0 -495
- data/vendor/crates/spikard-http/tests/grpc_server_streaming.rs +0 -974
- data/vendor/crates/spikard-http/tests/lifecycle_execution.rs +0 -1093
- data/vendor/crates/spikard-http/tests/middleware_stack_integration.rs +0 -389
- data/vendor/crates/spikard-http/tests/multipart_behavior.rs +0 -656
- data/vendor/crates/spikard-http/tests/request_extraction_full.rs +0 -513
- data/vendor/crates/spikard-http/tests/server_auth_middleware_behavior.rs +0 -328
- data/vendor/crates/spikard-http/tests/server_config_builder.rs +0 -314
- data/vendor/crates/spikard-http/tests/server_configured_router_behavior.rs +0 -200
- data/vendor/crates/spikard-http/tests/server_cors_preflight.rs +0 -83
- data/vendor/crates/spikard-http/tests/server_handler_wrappers.rs +0 -464
- data/vendor/crates/spikard-http/tests/server_method_router_additional_behavior.rs +0 -286
- data/vendor/crates/spikard-http/tests/server_method_router_coverage.rs +0 -118
- data/vendor/crates/spikard-http/tests/server_middleware_behavior.rs +0 -99
- data/vendor/crates/spikard-http/tests/server_middleware_branches.rs +0 -204
- data/vendor/crates/spikard-http/tests/server_openapi_jsonrpc_static.rs +0 -421
- data/vendor/crates/spikard-http/tests/server_router_behavior.rs +0 -121
- data/vendor/crates/spikard-http/tests/sse_behavior.rs +0 -620
- data/vendor/crates/spikard-http/tests/sse_full_behavior.rs +0 -584
- data/vendor/crates/spikard-http/tests/sse_handler_behavior.rs +0 -130
- data/vendor/crates/spikard-http/tests/test_client_requests.rs +0 -167
- data/vendor/crates/spikard-http/tests/testing_helpers.rs +0 -87
- data/vendor/crates/spikard-http/tests/testing_module_coverage.rs +0 -155
- data/vendor/crates/spikard-http/tests/urlencoded_content_type.rs +0 -82
- data/vendor/crates/spikard-http/tests/websocket_behavior.rs +0 -663
- data/vendor/crates/spikard-http/tests/websocket_full_behavior.rs +0 -440
- data/vendor/crates/spikard-http/tests/websocket_integration.rs +0 -150
- data/vendor/crates/spikard-rb/Cargo.toml +0 -68
- data/vendor/crates/spikard-rb/build.rs +0 -200
- data/vendor/crates/spikard-rb/src/background.rs +0 -63
- data/vendor/crates/spikard-rb/src/config/mod.rs +0 -5
- data/vendor/crates/spikard-rb/src/config/server_config.rs +0 -401
- data/vendor/crates/spikard-rb/src/conversion.rs +0 -688
- data/vendor/crates/spikard-rb/src/di/builder.rs +0 -100
- data/vendor/crates/spikard-rb/src/di/mod.rs +0 -375
- data/vendor/crates/spikard-rb/src/grpc/handler.rs +0 -834
- data/vendor/crates/spikard-rb/src/grpc/mod.rs +0 -13
- data/vendor/crates/spikard-rb/src/gvl.rs +0 -80
- data/vendor/crates/spikard-rb/src/handler.rs +0 -699
- data/vendor/crates/spikard-rb/src/integration/mod.rs +0 -3
- data/vendor/crates/spikard-rb/src/lib.rs +0 -2264
- data/vendor/crates/spikard-rb/src/lifecycle.rs +0 -303
- data/vendor/crates/spikard-rb/src/metadata/mod.rs +0 -5
- data/vendor/crates/spikard-rb/src/metadata/route_extraction.rs +0 -507
- data/vendor/crates/spikard-rb/src/request.rs +0 -439
- data/vendor/crates/spikard-rb/src/runtime/mod.rs +0 -5
- data/vendor/crates/spikard-rb/src/runtime/server_runner.rs +0 -344
- data/vendor/crates/spikard-rb/src/server.rs +0 -307
- data/vendor/crates/spikard-rb/src/sse.rs +0 -231
- data/vendor/crates/spikard-rb/src/testing/client.rs +0 -698
- data/vendor/crates/spikard-rb/src/testing/mod.rs +0 -7
- data/vendor/crates/spikard-rb/src/testing/sse.rs +0 -108
- data/vendor/crates/spikard-rb/src/testing/websocket.rs +0 -573
- data/vendor/crates/spikard-rb/src/websocket.rs +0 -475
- data/vendor/crates/spikard-rb-macros/Cargo.toml +0 -25
- data/vendor/crates/spikard-rb-macros/src/lib.rs +0 -51
|
@@ -1,430 +0,0 @@
|
|
|
1
|
-
//! Integration tests for gRPC bidirectional streaming
|
|
2
|
-
//!
|
|
3
|
-
//! Tests end-to-end bidirectional streaming functionality through GenericGrpcService
|
|
4
|
-
//! and grpc_routing, including:
|
|
5
|
-
//! - Echo service with bidirectional messages
|
|
6
|
-
//! - Message transformation during streaming
|
|
7
|
-
//! - Error handling in request and response streams
|
|
8
|
-
//! - Empty streams
|
|
9
|
-
//! - Large payloads (100+ messages in both directions)
|
|
10
|
-
//! - Message ordering preservation
|
|
11
|
-
//! - Concurrent bidirectional streaming requests
|
|
12
|
-
#![allow(
|
|
13
|
-
clippy::doc_markdown,
|
|
14
|
-
clippy::uninlined_format_args,
|
|
15
|
-
clippy::redundant_closure_for_method_calls,
|
|
16
|
-
reason = "Integration test for streaming with many test cases"
|
|
17
|
-
)]
|
|
18
|
-
|
|
19
|
-
use bytes::Bytes;
|
|
20
|
-
use futures_util::StreamExt;
|
|
21
|
-
use spikard_http::grpc::streaming::StreamingRequest;
|
|
22
|
-
use spikard_http::grpc::{GrpcHandler, GrpcHandlerResult, GrpcRequestData, RpcMode};
|
|
23
|
-
use std::future::Future;
|
|
24
|
-
use std::pin::Pin;
|
|
25
|
-
use tonic::metadata::MetadataMap;
|
|
26
|
-
|
|
27
|
-
// ============================================================================
|
|
28
|
-
// Test Handlers
|
|
29
|
-
// ============================================================================
|
|
30
|
-
|
|
31
|
-
/// Handler that echoes back messages from bidirectional stream
|
|
32
|
-
struct EchoBidiHandler;
|
|
33
|
-
|
|
34
|
-
impl GrpcHandler for EchoBidiHandler {
|
|
35
|
-
fn call(&self, _request: GrpcRequestData) -> Pin<Box<dyn Future<Output = GrpcHandlerResult> + Send>> {
|
|
36
|
-
Box::pin(async { Err(tonic::Status::unimplemented("Use call_bidi_stream for streaming")) })
|
|
37
|
-
}
|
|
38
|
-
|
|
39
|
-
fn service_name(&self) -> &'static str {
|
|
40
|
-
"integration.EchoService"
|
|
41
|
-
}
|
|
42
|
-
|
|
43
|
-
fn rpc_mode(&self) -> RpcMode {
|
|
44
|
-
RpcMode::BidirectionalStreaming
|
|
45
|
-
}
|
|
46
|
-
|
|
47
|
-
fn call_bidi_stream(
|
|
48
|
-
&self,
|
|
49
|
-
request: StreamingRequest,
|
|
50
|
-
) -> Pin<Box<dyn Future<Output = Result<spikard_http::grpc::streaming::MessageStream, tonic::Status>> + Send>> {
|
|
51
|
-
Box::pin(async move {
|
|
52
|
-
let mut messages = Vec::new();
|
|
53
|
-
let mut stream = request.message_stream;
|
|
54
|
-
|
|
55
|
-
while let Some(msg_result) = stream.next().await {
|
|
56
|
-
match msg_result {
|
|
57
|
-
Ok(msg) => messages.push(msg),
|
|
58
|
-
Err(e) => return Err(e),
|
|
59
|
-
}
|
|
60
|
-
}
|
|
61
|
-
|
|
62
|
-
Ok(spikard_http::grpc::streaming::message_stream_from_vec(messages))
|
|
63
|
-
})
|
|
64
|
-
}
|
|
65
|
-
}
|
|
66
|
-
|
|
67
|
-
/// Handler that transforms messages (uppercase) in bidirectional stream
|
|
68
|
-
struct TransformBidiHandler;
|
|
69
|
-
|
|
70
|
-
impl GrpcHandler for TransformBidiHandler {
|
|
71
|
-
fn call(&self, _request: GrpcRequestData) -> Pin<Box<dyn Future<Output = GrpcHandlerResult> + Send>> {
|
|
72
|
-
Box::pin(async { Err(tonic::Status::unimplemented("Use call_bidi_stream for streaming")) })
|
|
73
|
-
}
|
|
74
|
-
|
|
75
|
-
fn service_name(&self) -> &'static str {
|
|
76
|
-
"integration.TransformService"
|
|
77
|
-
}
|
|
78
|
-
|
|
79
|
-
fn rpc_mode(&self) -> RpcMode {
|
|
80
|
-
RpcMode::BidirectionalStreaming
|
|
81
|
-
}
|
|
82
|
-
|
|
83
|
-
fn call_bidi_stream(
|
|
84
|
-
&self,
|
|
85
|
-
request: StreamingRequest,
|
|
86
|
-
) -> Pin<Box<dyn Future<Output = Result<spikard_http::grpc::streaming::MessageStream, tonic::Status>> + Send>> {
|
|
87
|
-
Box::pin(async move {
|
|
88
|
-
let mut messages = Vec::new();
|
|
89
|
-
let mut stream = request.message_stream;
|
|
90
|
-
|
|
91
|
-
while let Some(msg_result) = stream.next().await {
|
|
92
|
-
match msg_result {
|
|
93
|
-
Ok(msg) => {
|
|
94
|
-
let text = String::from_utf8_lossy(&msg).to_uppercase();
|
|
95
|
-
messages.push(Bytes::from(text));
|
|
96
|
-
}
|
|
97
|
-
Err(e) => return Err(e),
|
|
98
|
-
}
|
|
99
|
-
}
|
|
100
|
-
|
|
101
|
-
Ok(spikard_http::grpc::streaming::message_stream_from_vec(messages))
|
|
102
|
-
})
|
|
103
|
-
}
|
|
104
|
-
}
|
|
105
|
-
|
|
106
|
-
// ============================================================================
|
|
107
|
-
// Integration Tests
|
|
108
|
-
// ============================================================================
|
|
109
|
-
|
|
110
|
-
#[tokio::test]
|
|
111
|
-
async fn test_integration_bidi_echo() {
|
|
112
|
-
let handler = EchoBidiHandler;
|
|
113
|
-
|
|
114
|
-
let messages = vec![Bytes::from("hello"), Bytes::from("world"), Bytes::from("test")];
|
|
115
|
-
let stream = spikard_http::grpc::streaming::message_stream_from_vec(messages.clone());
|
|
116
|
-
let request = StreamingRequest {
|
|
117
|
-
service_name: "integration.EchoService".to_string(),
|
|
118
|
-
method_name: "Echo".to_string(),
|
|
119
|
-
message_stream: stream,
|
|
120
|
-
metadata: MetadataMap::new(),
|
|
121
|
-
};
|
|
122
|
-
|
|
123
|
-
let result = handler.call_bidi_stream(request).await;
|
|
124
|
-
assert!(result.is_ok());
|
|
125
|
-
|
|
126
|
-
let mut response_stream = result.unwrap();
|
|
127
|
-
let mut responses = Vec::new();
|
|
128
|
-
|
|
129
|
-
while let Some(msg_result) = response_stream.next().await {
|
|
130
|
-
match msg_result {
|
|
131
|
-
Ok(msg) => responses.push(msg),
|
|
132
|
-
Err(_) => break,
|
|
133
|
-
}
|
|
134
|
-
}
|
|
135
|
-
|
|
136
|
-
assert_eq!(responses.len(), 3);
|
|
137
|
-
assert_eq!(responses, messages);
|
|
138
|
-
}
|
|
139
|
-
|
|
140
|
-
#[tokio::test]
|
|
141
|
-
async fn test_integration_bidi_transform() {
|
|
142
|
-
let handler = TransformBidiHandler;
|
|
143
|
-
|
|
144
|
-
let messages = vec![Bytes::from("hello"), Bytes::from("world")];
|
|
145
|
-
let stream = spikard_http::grpc::streaming::message_stream_from_vec(messages);
|
|
146
|
-
let request = StreamingRequest {
|
|
147
|
-
service_name: "integration.TransformService".to_string(),
|
|
148
|
-
method_name: "Transform".to_string(),
|
|
149
|
-
message_stream: stream,
|
|
150
|
-
metadata: MetadataMap::new(),
|
|
151
|
-
};
|
|
152
|
-
|
|
153
|
-
let result = handler.call_bidi_stream(request).await;
|
|
154
|
-
assert!(result.is_ok());
|
|
155
|
-
|
|
156
|
-
let mut response_stream = result.unwrap();
|
|
157
|
-
let mut responses = Vec::new();
|
|
158
|
-
|
|
159
|
-
while let Some(msg_result) = response_stream.next().await {
|
|
160
|
-
match msg_result {
|
|
161
|
-
Ok(msg) => responses.push(String::from_utf8_lossy(&msg).to_string()),
|
|
162
|
-
Err(_) => break,
|
|
163
|
-
}
|
|
164
|
-
}
|
|
165
|
-
|
|
166
|
-
assert_eq!(responses.len(), 2);
|
|
167
|
-
assert_eq!(responses[0], "HELLO");
|
|
168
|
-
assert_eq!(responses[1], "WORLD");
|
|
169
|
-
}
|
|
170
|
-
|
|
171
|
-
#[tokio::test]
|
|
172
|
-
async fn test_integration_bidi_empty_stream() {
|
|
173
|
-
let handler = EchoBidiHandler;
|
|
174
|
-
|
|
175
|
-
let messages: Vec<Bytes> = vec![];
|
|
176
|
-
let stream = spikard_http::grpc::streaming::message_stream_from_vec(messages);
|
|
177
|
-
let request = StreamingRequest {
|
|
178
|
-
service_name: "integration.EchoService".to_string(),
|
|
179
|
-
method_name: "Echo".to_string(),
|
|
180
|
-
message_stream: stream,
|
|
181
|
-
metadata: MetadataMap::new(),
|
|
182
|
-
};
|
|
183
|
-
|
|
184
|
-
let result = handler.call_bidi_stream(request).await;
|
|
185
|
-
assert!(result.is_ok());
|
|
186
|
-
|
|
187
|
-
let mut response_stream = result.unwrap();
|
|
188
|
-
let mut responses = Vec::new();
|
|
189
|
-
|
|
190
|
-
while let Some(msg_result) = response_stream.next().await {
|
|
191
|
-
match msg_result {
|
|
192
|
-
Ok(msg) => responses.push(msg),
|
|
193
|
-
Err(_) => break,
|
|
194
|
-
}
|
|
195
|
-
}
|
|
196
|
-
|
|
197
|
-
assert_eq!(responses.len(), 0);
|
|
198
|
-
}
|
|
199
|
-
|
|
200
|
-
#[tokio::test]
|
|
201
|
-
async fn test_integration_bidi_large_stream() {
|
|
202
|
-
let handler = EchoBidiHandler;
|
|
203
|
-
|
|
204
|
-
// Create a large stream of messages (100+)
|
|
205
|
-
let messages: Vec<Bytes> = (0..150).map(|i| Bytes::from(format!("msg_{}", i))).collect();
|
|
206
|
-
let stream = spikard_http::grpc::streaming::message_stream_from_vec(messages);
|
|
207
|
-
let request = StreamingRequest {
|
|
208
|
-
service_name: "integration.EchoService".to_string(),
|
|
209
|
-
method_name: "Echo".to_string(),
|
|
210
|
-
message_stream: stream,
|
|
211
|
-
metadata: MetadataMap::new(),
|
|
212
|
-
};
|
|
213
|
-
|
|
214
|
-
let result = handler.call_bidi_stream(request).await;
|
|
215
|
-
assert!(result.is_ok());
|
|
216
|
-
|
|
217
|
-
let mut response_stream = result.unwrap();
|
|
218
|
-
let mut responses = Vec::new();
|
|
219
|
-
|
|
220
|
-
while let Some(msg_result) = response_stream.next().await {
|
|
221
|
-
match msg_result {
|
|
222
|
-
Ok(msg) => responses.push(msg),
|
|
223
|
-
Err(_) => break,
|
|
224
|
-
}
|
|
225
|
-
}
|
|
226
|
-
|
|
227
|
-
assert_eq!(responses.len(), 150);
|
|
228
|
-
}
|
|
229
|
-
|
|
230
|
-
#[tokio::test]
|
|
231
|
-
async fn test_integration_bidi_metadata_propagation() {
|
|
232
|
-
let handler = EchoBidiHandler;
|
|
233
|
-
|
|
234
|
-
let mut metadata = MetadataMap::new();
|
|
235
|
-
metadata.insert("x-request-id", "bidi-001".parse().unwrap());
|
|
236
|
-
metadata.insert("x-custom", "value".parse().unwrap());
|
|
237
|
-
|
|
238
|
-
let messages = vec![Bytes::from("test")];
|
|
239
|
-
let stream = spikard_http::grpc::streaming::message_stream_from_vec(messages);
|
|
240
|
-
let request = StreamingRequest {
|
|
241
|
-
service_name: "integration.EchoService".to_string(),
|
|
242
|
-
method_name: "Echo".to_string(),
|
|
243
|
-
message_stream: stream,
|
|
244
|
-
metadata,
|
|
245
|
-
};
|
|
246
|
-
|
|
247
|
-
let result = handler.call_bidi_stream(request).await;
|
|
248
|
-
assert!(result.is_ok());
|
|
249
|
-
|
|
250
|
-
let mut response_stream = result.unwrap();
|
|
251
|
-
let mut count = 0;
|
|
252
|
-
|
|
253
|
-
while let Some(msg_result) = response_stream.next().await {
|
|
254
|
-
match msg_result {
|
|
255
|
-
Ok(_msg) => count += 1,
|
|
256
|
-
Err(_) => break,
|
|
257
|
-
}
|
|
258
|
-
}
|
|
259
|
-
|
|
260
|
-
assert_eq!(count, 1);
|
|
261
|
-
}
|
|
262
|
-
|
|
263
|
-
#[tokio::test]
|
|
264
|
-
async fn test_integration_bidi_concurrent_requests() {
|
|
265
|
-
/// Handler with request-specific ID
|
|
266
|
-
struct ConcurrentBidiHandler {
|
|
267
|
-
request_id: usize,
|
|
268
|
-
}
|
|
269
|
-
|
|
270
|
-
impl GrpcHandler for ConcurrentBidiHandler {
|
|
271
|
-
fn call(&self, _request: GrpcRequestData) -> Pin<Box<dyn Future<Output = GrpcHandlerResult> + Send>> {
|
|
272
|
-
Box::pin(async { Err(tonic::Status::unimplemented("Use call_bidi_stream for streaming")) })
|
|
273
|
-
}
|
|
274
|
-
|
|
275
|
-
fn service_name(&self) -> &'static str {
|
|
276
|
-
"integration.ConcurrentService"
|
|
277
|
-
}
|
|
278
|
-
|
|
279
|
-
fn rpc_mode(&self) -> RpcMode {
|
|
280
|
-
RpcMode::BidirectionalStreaming
|
|
281
|
-
}
|
|
282
|
-
|
|
283
|
-
fn call_bidi_stream(
|
|
284
|
-
&self,
|
|
285
|
-
request: StreamingRequest,
|
|
286
|
-
) -> Pin<Box<dyn Future<Output = Result<spikard_http::grpc::streaming::MessageStream, tonic::Status>> + Send>>
|
|
287
|
-
{
|
|
288
|
-
let request_id = self.request_id;
|
|
289
|
-
Box::pin(async move {
|
|
290
|
-
let mut stream = request.message_stream;
|
|
291
|
-
let mut count = 0;
|
|
292
|
-
let mut responses = Vec::new();
|
|
293
|
-
|
|
294
|
-
while let Some(msg_result) = stream.next().await {
|
|
295
|
-
match msg_result {
|
|
296
|
-
Ok(_msg) => {
|
|
297
|
-
count += 1;
|
|
298
|
-
responses.push(Bytes::from(format!("req_{}:{}", request_id, count)));
|
|
299
|
-
}
|
|
300
|
-
Err(e) => return Err(e),
|
|
301
|
-
}
|
|
302
|
-
}
|
|
303
|
-
|
|
304
|
-
Ok(spikard_http::grpc::streaming::message_stream_from_vec(responses))
|
|
305
|
-
})
|
|
306
|
-
}
|
|
307
|
-
}
|
|
308
|
-
|
|
309
|
-
let handler1 = ConcurrentBidiHandler { request_id: 1 };
|
|
310
|
-
let handler2 = ConcurrentBidiHandler { request_id: 2 };
|
|
311
|
-
let handler3 = ConcurrentBidiHandler { request_id: 3 };
|
|
312
|
-
|
|
313
|
-
let task1 = tokio::spawn(async move {
|
|
314
|
-
let messages = vec![Bytes::from("a"), Bytes::from("b")];
|
|
315
|
-
let stream = spikard_http::grpc::streaming::message_stream_from_vec(messages);
|
|
316
|
-
let request = StreamingRequest {
|
|
317
|
-
service_name: "integration.ConcurrentService".to_string(),
|
|
318
|
-
method_name: "Concurrent".to_string(),
|
|
319
|
-
message_stream: stream,
|
|
320
|
-
metadata: MetadataMap::new(),
|
|
321
|
-
};
|
|
322
|
-
handler1.call_bidi_stream(request).await
|
|
323
|
-
});
|
|
324
|
-
|
|
325
|
-
let task2 = tokio::spawn(async move {
|
|
326
|
-
let messages = vec![Bytes::from("x"), Bytes::from("y"), Bytes::from("z")];
|
|
327
|
-
let stream = spikard_http::grpc::streaming::message_stream_from_vec(messages);
|
|
328
|
-
let request = StreamingRequest {
|
|
329
|
-
service_name: "integration.ConcurrentService".to_string(),
|
|
330
|
-
method_name: "Concurrent".to_string(),
|
|
331
|
-
message_stream: stream,
|
|
332
|
-
metadata: MetadataMap::new(),
|
|
333
|
-
};
|
|
334
|
-
handler2.call_bidi_stream(request).await
|
|
335
|
-
});
|
|
336
|
-
|
|
337
|
-
let task3 = tokio::spawn(async move {
|
|
338
|
-
let messages = vec![Bytes::from("single")];
|
|
339
|
-
let stream = spikard_http::grpc::streaming::message_stream_from_vec(messages);
|
|
340
|
-
let request = StreamingRequest {
|
|
341
|
-
service_name: "integration.ConcurrentService".to_string(),
|
|
342
|
-
method_name: "Concurrent".to_string(),
|
|
343
|
-
message_stream: stream,
|
|
344
|
-
metadata: MetadataMap::new(),
|
|
345
|
-
};
|
|
346
|
-
handler3.call_bidi_stream(request).await
|
|
347
|
-
});
|
|
348
|
-
|
|
349
|
-
let result1 = task1.await.unwrap();
|
|
350
|
-
let result2 = task2.await.unwrap();
|
|
351
|
-
let result3 = task3.await.unwrap();
|
|
352
|
-
|
|
353
|
-
assert!(result1.is_ok());
|
|
354
|
-
assert!(result2.is_ok());
|
|
355
|
-
assert!(result3.is_ok());
|
|
356
|
-
|
|
357
|
-
// Verify response counts
|
|
358
|
-
let mut responses1 = Vec::new();
|
|
359
|
-
if let Ok(mut stream) = result1 {
|
|
360
|
-
while let Some(msg_result) = stream.next().await {
|
|
361
|
-
if let Ok(msg) = msg_result {
|
|
362
|
-
responses1.push(msg);
|
|
363
|
-
}
|
|
364
|
-
}
|
|
365
|
-
}
|
|
366
|
-
assert_eq!(responses1.len(), 2);
|
|
367
|
-
}
|
|
368
|
-
|
|
369
|
-
#[tokio::test]
|
|
370
|
-
async fn test_integration_bidi_ordering_preserved() {
|
|
371
|
-
let handler = EchoBidiHandler;
|
|
372
|
-
|
|
373
|
-
// Verify that message ordering is preserved through bidi stream
|
|
374
|
-
let messages: Vec<Bytes> = (0..10).map(|i| Bytes::from(format!("msg{:02}", i))).collect();
|
|
375
|
-
let stream = spikard_http::grpc::streaming::message_stream_from_vec(messages.clone());
|
|
376
|
-
let request = StreamingRequest {
|
|
377
|
-
service_name: "integration.EchoService".to_string(),
|
|
378
|
-
method_name: "Echo".to_string(),
|
|
379
|
-
message_stream: stream,
|
|
380
|
-
metadata: MetadataMap::new(),
|
|
381
|
-
};
|
|
382
|
-
|
|
383
|
-
let result = handler.call_bidi_stream(request).await;
|
|
384
|
-
assert!(result.is_ok());
|
|
385
|
-
|
|
386
|
-
let mut response_stream = result.unwrap();
|
|
387
|
-
let mut responses = Vec::new();
|
|
388
|
-
|
|
389
|
-
while let Some(msg_result) = response_stream.next().await {
|
|
390
|
-
match msg_result {
|
|
391
|
-
Ok(msg) => responses.push(msg),
|
|
392
|
-
Err(_) => break,
|
|
393
|
-
}
|
|
394
|
-
}
|
|
395
|
-
|
|
396
|
-
assert_eq!(responses.len(), 10);
|
|
397
|
-
for (i, resp) in responses.iter().enumerate() {
|
|
398
|
-
assert_eq!(resp, &Bytes::from(format!("msg{:02}", i)));
|
|
399
|
-
}
|
|
400
|
-
}
|
|
401
|
-
|
|
402
|
-
#[tokio::test]
|
|
403
|
-
async fn test_integration_bidi_single_message() {
|
|
404
|
-
let handler = EchoBidiHandler;
|
|
405
|
-
|
|
406
|
-
let messages = vec![Bytes::from("single_message")];
|
|
407
|
-
let stream = spikard_http::grpc::streaming::message_stream_from_vec(messages.clone());
|
|
408
|
-
let request = StreamingRequest {
|
|
409
|
-
service_name: "integration.EchoService".to_string(),
|
|
410
|
-
method_name: "Echo".to_string(),
|
|
411
|
-
message_stream: stream,
|
|
412
|
-
metadata: MetadataMap::new(),
|
|
413
|
-
};
|
|
414
|
-
|
|
415
|
-
let result = handler.call_bidi_stream(request).await;
|
|
416
|
-
assert!(result.is_ok());
|
|
417
|
-
|
|
418
|
-
let mut response_stream = result.unwrap();
|
|
419
|
-
let mut responses = Vec::new();
|
|
420
|
-
|
|
421
|
-
while let Some(msg_result) = response_stream.next().await {
|
|
422
|
-
match msg_result {
|
|
423
|
-
Ok(msg) => responses.push(msg),
|
|
424
|
-
Err(_) => break,
|
|
425
|
-
}
|
|
426
|
-
}
|
|
427
|
-
|
|
428
|
-
assert_eq!(responses.len(), 1);
|
|
429
|
-
assert_eq!(responses[0], messages[0]);
|
|
430
|
-
}
|