blob: 06a7029a6d3c78ff702c2a49cd4403780bb3c24a [file] [log] [blame]
Yang Gaoa5e20d32015-03-25 09:55:20 -07001/*
2 *
3 * Copyright 2015, Google Inc.
4 * All rights reserved.
5 *
6 * Redistribution and use in source and binary forms, with or without
7 * modification, are permitted provided that the following conditions are
8 * met:
9 *
10 * * Redistributions of source code must retain the above copyright
11 * notice, this list of conditions and the following disclaimer.
12 * * Redistributions in binary form must reproduce the above
13 * copyright notice, this list of conditions and the following disclaimer
14 * in the documentation and/or other materials provided with the
15 * distribution.
16 * * Neither the name of Google Inc. nor the names of its
17 * contributors may be used to endorse or promote products derived from
18 * this software without specific prior written permission.
19 *
20 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
21 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
22 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
23 * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
24 * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
25 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
26 * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
27 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
28 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
29 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
30 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
31 *
32 */
33
Yang Gaoced2b892015-03-26 22:59:54 -070034/*
35 A command line tool to talk to any grpc server.
36 Example of talking to grpc interop server:
37 1. Prepare request binary file:
38 a. create a text file input.txt, containing the following:
39 response_size: 10
40 payload: {
41 body: "hello world"
42 }
43 b. under grpc/ run
Yang Gao102eccb2015-06-16 00:43:25 -070044 protoc --proto_path=test/proto/ \
45 --encode=grpc.testing.SimpleRequest test/proto/messages.proto \
Yang Gaoced2b892015-03-26 22:59:54 -070046 < input.txt > input.bin
47 2. Start a server
48 make interop_server && bins/opt/interop_server --port=50051
49 3. Run the tool
50 make grpc_cli && bins/opt/grpc_cli call localhost:50051 \
51 /grpc.testing.TestService/UnaryCall --enable_ssl=false \
52 --input_binary_file=input.bin --output_binary_file=output.bin
53 4. Decode response
Yang Gao102eccb2015-06-16 00:43:25 -070054 protoc --proto_path=test/proto/ \
55 --decode=grpc.testing.SimpleResponse test/proto/messages.proto \
Yang Gaoced2b892015-03-26 22:59:54 -070056 < output.bin > output.txt
57 5. Now the text form of response should be in output.txt
Yang Gao102eccb2015-06-16 00:43:25 -070058 Optionally, metadata can be passed to server via flag --metadata, e.g.
59 --metadata="MyHeaderKey1:Value1:MyHeaderKey2:Value2"
Yang Gaoced2b892015-03-26 22:59:54 -070060*/
Yang Gaoa5e20d32015-03-25 09:55:20 -070061
Yang Gaoced2b892015-03-26 22:59:54 -070062#include <fstream>
63#include <iostream>
64#include <sstream>
65
Yang Gaoa5e20d32015-03-25 09:55:20 -070066#include <gflags/gflags.h>
Yang Gaob946b5e2015-03-27 13:20:59 -070067#include "test/cpp/util/cli_call.h"
Yang Gao103837e2015-04-15 15:23:54 -070068#include "test/cpp/util/test_config.h"
Yang Gaoa5e20d32015-03-25 09:55:20 -070069#include <grpc++/channel_arguments.h>
70#include <grpc++/channel_interface.h>
Yang Gaoa5e20d32015-03-25 09:55:20 -070071#include <grpc++/create_channel.h>
72#include <grpc++/credentials.h>
Yang Gaoa5e20d32015-03-25 09:55:20 -070073
74#include <grpc/grpc.h>
Yang Gaoa5e20d32015-03-25 09:55:20 -070075
Yang Gaoced2b892015-03-26 22:59:54 -070076DEFINE_bool(enable_ssl, true, "Whether to use ssl/tls.");
77DEFINE_bool(use_auth, false, "Whether to create default google credentials.");
78DEFINE_string(input_binary_file, "",
79 "Path to input file containing serialized request.");
80DEFINE_string(output_binary_file, "output.bin",
81 "Path to output file to write serialized response.");
Yang Gao102eccb2015-06-16 00:43:25 -070082DEFINE_string(metadata, "",
83 "Metadata to send to server, in the form of key1:val1:key2:val2");
84
85void ParseMetadataFlag(
86 std::multimap<grpc::string, grpc::string>* client_metadata) {
87 if (FLAGS_metadata.empty()) {
88 return;
89 }
90 std::vector<grpc::string> fields;
91 grpc::string delim(":");
92 size_t cur, next = -1;
93 do {
94 cur = next + 1;
95 next = FLAGS_metadata.find_first_of(delim, cur);
96 fields.push_back(FLAGS_metadata.substr(cur, next - cur));
97 } while (next != grpc::string::npos);
98 if (fields.size() % 2) {
99 std::cout << "Failed to parse metadata flag" << std::endl;
100 exit(1);
101 }
102 for (size_t i = 0; i < fields.size(); i += 2) {
103 client_metadata->insert(
104 std::pair<grpc::string, grpc::string>(fields[i], fields[i + 1]));
105 }
106}
107
108void PrintMetadata(const std::multimap<grpc::string, grpc::string>& m,
109 const grpc::string& message) {
110 if (m.empty()) {
111 return;
112 }
113 std::cout << message << std::endl;
114 for (std::multimap<grpc::string, grpc::string>::const_iterator iter =
115 m.begin();
116 iter != m.end(); ++iter) {
117 std::cout << iter->first << " : " << iter->second << std::endl;
118 }
119}
Yang Gaoa5e20d32015-03-25 09:55:20 -0700120
Yang Gaoa5e20d32015-03-25 09:55:20 -0700121int main(int argc, char** argv) {
Yang Gao103837e2015-04-15 15:23:54 -0700122 grpc::testing::InitTest(&argc, &argv, true);
Yang Gaoa5e20d32015-03-25 09:55:20 -0700123
Yang Gaoced2b892015-03-26 22:59:54 -0700124 if (argc < 4 || grpc::string(argv[1]) != "call") {
Yang Gaoa5e20d32015-03-25 09:55:20 -0700125 std::cout << "Usage: grpc_cli call server_host:port full_method_string\n"
126 << "Example: grpc_cli call service.googleapis.com "
Yang Gaoced2b892015-03-26 22:59:54 -0700127 << "/grpc.testing.TestService/UnaryCall "
128 << "--input_binary_file=input.bin --output_binary_file=output.bin"
129 << std::endl;
Yang Gaoa5e20d32015-03-25 09:55:20 -0700130 }
131 grpc::string server_address(argv[2]);
Yang Gaob946b5e2015-03-27 13:20:59 -0700132 // TODO(yangg) basic check of method string
Yang Gao166f9d02015-03-26 23:06:45 -0700133 grpc::string method(argv[3]);
Yang Gaoa5e20d32015-03-25 09:55:20 -0700134
Yang Gaoced2b892015-03-26 22:59:54 -0700135 if (FLAGS_input_binary_file.empty()) {
136 std::cout << "Missing --input_binary_file for serialized request."
137 << std::endl;
138 return 1;
139 }
140 std::cout << "connecting to " << server_address << std::endl;
141
Yang Gaob946b5e2015-03-27 13:20:59 -0700142 std::ifstream input_file(FLAGS_input_binary_file,
143 std::ios::in | std::ios::binary);
144 std::stringstream input_stream;
145 input_stream << input_file.rdbuf();
Yang Gaoced2b892015-03-26 22:59:54 -0700146
Yang Gaoa8938922015-05-14 11:51:07 -0700147 std::shared_ptr<grpc::Credentials> creds;
Yang Gaoced2b892015-03-26 22:59:54 -0700148 if (!FLAGS_enable_ssl) {
149 creds = grpc::InsecureCredentials();
150 } else {
151 if (FLAGS_use_auth) {
152 creds = grpc::GoogleDefaultCredentials();
153 } else {
154 creds = grpc::SslCredentials(grpc::SslCredentialsOptions());
155 }
156 }
Yang Gaoa5e20d32015-03-25 09:55:20 -0700157 std::shared_ptr<grpc::ChannelInterface> channel =
158 grpc::CreateChannel(server_address, creds, grpc::ChannelArguments());
159
Yang Gaob946b5e2015-03-27 13:20:59 -0700160 grpc::string response;
Yang Gao102eccb2015-06-16 00:43:25 -0700161 std::multimap<grpc::string, grpc::string> client_metadata,
162 server_initial_metadata, server_trailing_metadata;
163 ParseMetadataFlag(&client_metadata);
164 PrintMetadata(client_metadata, "Sending client initial metadata:");
165 grpc::Status s = grpc::testing::CliCall::Call(
166 channel, method, input_stream.str(), &response, client_metadata,
167 &server_initial_metadata, &server_trailing_metadata);
168 PrintMetadata(server_initial_metadata,
169 "Received initial metadata from server:");
170 PrintMetadata(server_trailing_metadata,
171 "Received trailing metadata from server:");
172 if (s.IsOk()) {
173 std::cout << "Rpc succeeded with OK status" << std::endl;
174 if (!response.empty()) {
175 std::ofstream output_file(FLAGS_output_binary_file,
176 std::ios::trunc | std::ios::binary);
177 output_file << response;
178 }
179 } else {
180 std::cout << "Rpc failed with status code " << s.code() << " error message "
181 << s.details() << std::endl;
Yang Gaob946b5e2015-03-27 13:20:59 -0700182 }
Yang Gaoa5e20d32015-03-25 09:55:20 -0700183
Yang Gaoa5e20d32015-03-25 09:55:20 -0700184 return 0;
185}