/* * * Copyright 2017 gRPC authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. * */ #import #include #include #include #include #include #include #include #include #include #include #include #include #include "src/core/lib/gprpp/thd.h" #include "test/core/util/port.h" #include "test/core/util/test_config.h" using std::chrono::system_clock; using namespace grpc; void* tag(int i) { return (void*)(intptr_t)i; } static grpc_slice merge_slices(grpc_slice* slices, size_t nslices) { size_t i; size_t len = 0; uint8_t* cursor; grpc_slice out; for (i = 0; i < nslices; i++) { len += GRPC_SLICE_LENGTH(slices[i]); } out = grpc_slice_malloc(len); cursor = GRPC_SLICE_START_PTR(out); for (i = 0; i < nslices; i++) { memcpy(cursor, GRPC_SLICE_START_PTR(slices[i]), GRPC_SLICE_LENGTH(slices[i])); cursor += GRPC_SLICE_LENGTH(slices[i]); } return out; } int byte_buffer_eq_string(ByteBuffer* bb, const char* str) { int res; std::vector slices; bb->Dump(&slices); grpc_slice* c_slices = new grpc_slice[slices.size()]; for (int i = 0; i < slices.size(); i++) { c_slices[i] = slices[i].c_slice(); } grpc_slice a = merge_slices(c_slices, slices.size()); grpc_slice b = grpc_slice_from_copied_string(str); res = (GRPC_SLICE_LENGTH(a) == GRPC_SLICE_LENGTH(b)) && (0 == memcmp(GRPC_SLICE_START_PTR(a), GRPC_SLICE_START_PTR(b), GRPC_SLICE_LENGTH(a))); grpc_slice_unref(a); grpc_slice_unref(b); for (int i = 0; i < slices.size(); i++) { grpc_slice_unref(c_slices[i]); } delete[] c_slices; return res; } @interface GenericTest : XCTestCase @end @implementation GenericTest { std::string server_host_; CompletionQueue cli_cq_; std::unique_ptr srv_cq_; std::unique_ptr generic_stub_; std::unique_ptr server_; AsyncGenericService generic_service_; std::ostringstream server_address_; } - (void)verify_ok:(grpc::CompletionQueue*)cq i:(int)i expect_ok:(bool)expect_ok { bool ok; void* got_tag; XCTAssertTrue(cq->Next(&got_tag, &ok)); XCTAssertEqual(expect_ok, ok); XCTAssertEqual(tag(i), got_tag); } - (void)server_ok:(int)i { [self verify_ok:srv_cq_.get() i:i expect_ok:true]; } - (void)client_ok:(int)i { [self verify_ok:&cli_cq_ i:i expect_ok:true]; } - (void)server_fail:(int)i { [self verify_ok:srv_cq_.get() i:i expect_ok:false]; } - (void)client_fail:(int)i { [self verify_ok:&cli_cq_ i:i expect_ok:false]; } - (void)setUp { [super setUp]; server_host_ = "localhost"; int port = grpc_pick_unused_port_or_die(); server_address_ << server_host_ << ":" << port; // Setup server ServerBuilder builder; builder.AddListeningPort(server_address_.str(), InsecureServerCredentials()); builder.RegisterAsyncGenericService(&generic_service_); // Include a second call to RegisterAsyncGenericService to make sure that // we get an error in the log, since it is not allowed to have 2 async // generic services builder.RegisterAsyncGenericService(&generic_service_); srv_cq_ = builder.AddCompletionQueue(); server_ = builder.BuildAndStart(); } - (void)tearDown { // Put teardown code here. This method is called after the invocation of each test method in the // class. server_->Shutdown(); void* ignored_tag; bool ignored_ok; cli_cq_.Shutdown(); srv_cq_->Shutdown(); while (cli_cq_.Next(&ignored_tag, &ignored_ok)) ; while (srv_cq_->Next(&ignored_tag, &ignored_ok)) ; [super tearDown]; } - (void)ResetStub { std::shared_ptr channel = CreateChannel(server_address_.str(), InsecureChannelCredentials()); generic_stub_.reset(new GenericStub(channel)); } - (void)SendRpc:(int)num_rpcs { [self SendRpc:num_rpcs check_deadline:false deadline:gpr_inf_future(GPR_CLOCK_MONOTONIC)]; } - (void)SendRpc:(int)num_rpcs check_deadline:(bool)check_deadline deadline:(gpr_timespec)deadline { const std::string kMethodName("/grpc.cpp.test.util.EchoTestService/Echo"); for (int i = 0; i < num_rpcs; i++) { Status recv_status; ClientContext cli_ctx; GenericServerContext srv_ctx; GenericServerAsyncReaderWriter stream(&srv_ctx); // The string needs to be long enough to test heap-based slice. /*send_request.set_message("Hello world. Hello world. Hello world.");*/ if (check_deadline) { cli_ctx.set_deadline(deadline); } std::unique_ptr call = generic_stub_->Call(&cli_ctx, kMethodName, &cli_cq_, tag(1)); [self client_ok:1]; Slice send_slice = Slice("hello world", 11); std::unique_ptr send_buffer = std::unique_ptr(new ByteBuffer(&send_slice, 1)); call->Write(*send_buffer, tag(2)); // Send ByteBuffer can be destroyed after calling Write. send_buffer.reset(); [self client_ok:2]; call->WritesDone(tag(3)); [self client_ok:3]; generic_service_.RequestCall(&srv_ctx, &stream, srv_cq_.get(), srv_cq_.get(), tag(4)); [self verify_ok:srv_cq_.get() i:4 expect_ok:true]; XCTAssertEqual(server_host_, srv_ctx.host().substr(0, server_host_.length())); XCTAssertEqual(kMethodName, srv_ctx.method()); if (check_deadline) { XCTAssertTrue(gpr_time_similar(deadline, srv_ctx.raw_deadline(), gpr_time_from_millis(1000, GPR_TIMESPAN))); } ByteBuffer recv_buffer; stream.Read(&recv_buffer, tag(5)); [self server_ok:5]; XCTAssertTrue(byte_buffer_eq_string(&recv_buffer, "hello world")); send_buffer = std::unique_ptr(new ByteBuffer(recv_buffer)); stream.Write(*send_buffer, tag(6)); send_buffer.reset(); [self server_ok:6]; stream.Finish(Status::OK, tag(7)); [self server_ok:7]; recv_buffer.Clear(); call->Read(&recv_buffer, tag(8)); [self client_ok:8]; XCTAssertTrue(byte_buffer_eq_string(&recv_buffer, "hello world")); call->Finish(&recv_status, tag(9)); [self client_ok:9]; XCTAssertTrue(recv_status.ok()); } } - (void)testSimpleRpc { [self ResetStub]; [self SendRpc:1]; } - (void)testSequentialRpcs { [self ResetStub]; [self SendRpc:10]; } + (void)setUp { grpc_test_init(0, NULL); } @end