1*cc02d7e2SAndroid Build Coastguard Worker // Copyright 2023 gRPC authors.
2*cc02d7e2SAndroid Build Coastguard Worker //
3*cc02d7e2SAndroid Build Coastguard Worker // Licensed under the Apache License, Version 2.0 (the "License");
4*cc02d7e2SAndroid Build Coastguard Worker // you may not use this file except in compliance with the License.
5*cc02d7e2SAndroid Build Coastguard Worker // You may obtain a copy of the License at
6*cc02d7e2SAndroid Build Coastguard Worker //
7*cc02d7e2SAndroid Build Coastguard Worker // http://www.apache.org/licenses/LICENSE-2.0
8*cc02d7e2SAndroid Build Coastguard Worker //
9*cc02d7e2SAndroid Build Coastguard Worker // Unless required by applicable law or agreed to in writing, software
10*cc02d7e2SAndroid Build Coastguard Worker // distributed under the License is distributed on an "AS IS" BASIS,
11*cc02d7e2SAndroid Build Coastguard Worker // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12*cc02d7e2SAndroid Build Coastguard Worker // See the License for the specific language governing permissions and
13*cc02d7e2SAndroid Build Coastguard Worker // limitations under the License.
14*cc02d7e2SAndroid Build Coastguard Worker
15*cc02d7e2SAndroid Build Coastguard Worker #include <condition_variable>
16*cc02d7e2SAndroid Build Coastguard Worker #include <iostream>
17*cc02d7e2SAndroid Build Coastguard Worker #include <memory>
18*cc02d7e2SAndroid Build Coastguard Worker #include <mutex>
19*cc02d7e2SAndroid Build Coastguard Worker #include <string>
20*cc02d7e2SAndroid Build Coastguard Worker #include <vector>
21*cc02d7e2SAndroid Build Coastguard Worker
22*cc02d7e2SAndroid Build Coastguard Worker #include "absl/flags/flag.h"
23*cc02d7e2SAndroid Build Coastguard Worker #include "absl/flags/parse.h"
24*cc02d7e2SAndroid Build Coastguard Worker #include "absl/strings/str_cat.h"
25*cc02d7e2SAndroid Build Coastguard Worker
26*cc02d7e2SAndroid Build Coastguard Worker #include <grpcpp/grpcpp.h>
27*cc02d7e2SAndroid Build Coastguard Worker
28*cc02d7e2SAndroid Build Coastguard Worker #ifdef BAZEL_BUILD
29*cc02d7e2SAndroid Build Coastguard Worker #include "examples/protos/helloworld.grpc.pb.h"
30*cc02d7e2SAndroid Build Coastguard Worker #else
31*cc02d7e2SAndroid Build Coastguard Worker #include "helloworld.grpc.pb.h"
32*cc02d7e2SAndroid Build Coastguard Worker #endif
33*cc02d7e2SAndroid Build Coastguard Worker
34*cc02d7e2SAndroid Build Coastguard Worker ABSL_FLAG(std::string, target, "localhost:50051", "Server address");
35*cc02d7e2SAndroid Build Coastguard Worker
36*cc02d7e2SAndroid Build Coastguard Worker using grpc::Channel;
37*cc02d7e2SAndroid Build Coastguard Worker using grpc::ClientContext;
38*cc02d7e2SAndroid Build Coastguard Worker using grpc::Status;
39*cc02d7e2SAndroid Build Coastguard Worker using grpc::StatusCode;
40*cc02d7e2SAndroid Build Coastguard Worker using helloworld::Greeter;
41*cc02d7e2SAndroid Build Coastguard Worker using helloworld::HelloReply;
42*cc02d7e2SAndroid Build Coastguard Worker using helloworld::HelloRequest;
43*cc02d7e2SAndroid Build Coastguard Worker
44*cc02d7e2SAndroid Build Coastguard Worker // Requests each key in the vector and displays the key and its corresponding
45*cc02d7e2SAndroid Build Coastguard Worker // value as a pair.
46*cc02d7e2SAndroid Build Coastguard Worker class KeyValueStoreClient
47*cc02d7e2SAndroid Build Coastguard Worker : public grpc::ClientBidiReactor<HelloRequest, HelloReply> {
48*cc02d7e2SAndroid Build Coastguard Worker public:
KeyValueStoreClient(std::shared_ptr<Channel> channel)49*cc02d7e2SAndroid Build Coastguard Worker KeyValueStoreClient(std::shared_ptr<Channel> channel)
50*cc02d7e2SAndroid Build Coastguard Worker : stub_(Greeter::NewStub(channel)) {
51*cc02d7e2SAndroid Build Coastguard Worker stub_->async()->SayHelloBidiStream(&context_, this);
52*cc02d7e2SAndroid Build Coastguard Worker request_.set_name("Begin");
53*cc02d7e2SAndroid Build Coastguard Worker StartWrite(&request_);
54*cc02d7e2SAndroid Build Coastguard Worker StartCall();
55*cc02d7e2SAndroid Build Coastguard Worker }
56*cc02d7e2SAndroid Build Coastguard Worker
OnReadDone(bool ok)57*cc02d7e2SAndroid Build Coastguard Worker void OnReadDone(bool ok) override {
58*cc02d7e2SAndroid Build Coastguard Worker if (ok) {
59*cc02d7e2SAndroid Build Coastguard Worker std::cout << request_.name() << " : " << response_.message() << std::endl;
60*cc02d7e2SAndroid Build Coastguard Worker if (++counter_ < 10) {
61*cc02d7e2SAndroid Build Coastguard Worker request_.set_name(absl::StrCat("Count ", counter_));
62*cc02d7e2SAndroid Build Coastguard Worker StartWrite(&request_);
63*cc02d7e2SAndroid Build Coastguard Worker } else {
64*cc02d7e2SAndroid Build Coastguard Worker // Cancel after sending 10 messages
65*cc02d7e2SAndroid Build Coastguard Worker context_.TryCancel();
66*cc02d7e2SAndroid Build Coastguard Worker }
67*cc02d7e2SAndroid Build Coastguard Worker }
68*cc02d7e2SAndroid Build Coastguard Worker }
69*cc02d7e2SAndroid Build Coastguard Worker
OnWriteDone(bool ok)70*cc02d7e2SAndroid Build Coastguard Worker void OnWriteDone(bool ok) override {
71*cc02d7e2SAndroid Build Coastguard Worker if (ok) {
72*cc02d7e2SAndroid Build Coastguard Worker StartRead(&response_);
73*cc02d7e2SAndroid Build Coastguard Worker }
74*cc02d7e2SAndroid Build Coastguard Worker }
75*cc02d7e2SAndroid Build Coastguard Worker
OnDone(const grpc::Status & status)76*cc02d7e2SAndroid Build Coastguard Worker void OnDone(const grpc::Status& status) override {
77*cc02d7e2SAndroid Build Coastguard Worker if (!status.ok()) {
78*cc02d7e2SAndroid Build Coastguard Worker if (status.error_code() == StatusCode::CANCELLED) {
79*cc02d7e2SAndroid Build Coastguard Worker // Eventually client will know here that call is cancelled.
80*cc02d7e2SAndroid Build Coastguard Worker std::cout << "RPC Cancelled!" << std::endl;
81*cc02d7e2SAndroid Build Coastguard Worker } else {
82*cc02d7e2SAndroid Build Coastguard Worker std::cout << "RPC Failed: " << status.error_code() << ": "
83*cc02d7e2SAndroid Build Coastguard Worker << status.error_message() << std::endl;
84*cc02d7e2SAndroid Build Coastguard Worker }
85*cc02d7e2SAndroid Build Coastguard Worker }
86*cc02d7e2SAndroid Build Coastguard Worker std::unique_lock<std::mutex> l(mu_);
87*cc02d7e2SAndroid Build Coastguard Worker done_ = true;
88*cc02d7e2SAndroid Build Coastguard Worker cv_.notify_all();
89*cc02d7e2SAndroid Build Coastguard Worker }
90*cc02d7e2SAndroid Build Coastguard Worker
Await()91*cc02d7e2SAndroid Build Coastguard Worker void Await() {
92*cc02d7e2SAndroid Build Coastguard Worker std::unique_lock<std::mutex> l(mu_);
93*cc02d7e2SAndroid Build Coastguard Worker while (!done_) {
94*cc02d7e2SAndroid Build Coastguard Worker cv_.wait(l);
95*cc02d7e2SAndroid Build Coastguard Worker }
96*cc02d7e2SAndroid Build Coastguard Worker }
97*cc02d7e2SAndroid Build Coastguard Worker
98*cc02d7e2SAndroid Build Coastguard Worker private:
99*cc02d7e2SAndroid Build Coastguard Worker std::unique_ptr<Greeter::Stub> stub_;
100*cc02d7e2SAndroid Build Coastguard Worker size_t counter_ = 0;
101*cc02d7e2SAndroid Build Coastguard Worker ClientContext context_;
102*cc02d7e2SAndroid Build Coastguard Worker bool done_ = false;
103*cc02d7e2SAndroid Build Coastguard Worker HelloRequest request_;
104*cc02d7e2SAndroid Build Coastguard Worker HelloReply response_;
105*cc02d7e2SAndroid Build Coastguard Worker std::mutex mu_;
106*cc02d7e2SAndroid Build Coastguard Worker std::condition_variable cv_;
107*cc02d7e2SAndroid Build Coastguard Worker };
108*cc02d7e2SAndroid Build Coastguard Worker
main(int argc,char ** argv)109*cc02d7e2SAndroid Build Coastguard Worker int main(int argc, char** argv) {
110*cc02d7e2SAndroid Build Coastguard Worker absl::ParseCommandLine(argc, argv);
111*cc02d7e2SAndroid Build Coastguard Worker // Instantiate the client. It requires a channel, out of which the actual RPCs
112*cc02d7e2SAndroid Build Coastguard Worker // are created. This channel models a connection to an endpoint specified by
113*cc02d7e2SAndroid Build Coastguard Worker // the argument "--target=" which is the only expected argument.
114*cc02d7e2SAndroid Build Coastguard Worker std::string target_str = absl::GetFlag(FLAGS_target);
115*cc02d7e2SAndroid Build Coastguard Worker KeyValueStoreClient client(
116*cc02d7e2SAndroid Build Coastguard Worker grpc::CreateChannel(target_str, grpc::InsecureChannelCredentials()));
117*cc02d7e2SAndroid Build Coastguard Worker client.Await();
118*cc02d7e2SAndroid Build Coastguard Worker return 0;
119*cc02d7e2SAndroid Build Coastguard Worker }
120