|
|
|
//
|
|
|
|
// Copyright 2022 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.
|
|
|
|
//
|
|
|
|
|
|
|
|
#include "src/cpp/ext/gcp/observability_logging_sink.h"
|
|
|
|
|
|
|
|
#include "absl/strings/escaping.h"
|
|
|
|
#include "absl/strings/str_format.h"
|
|
|
|
#include "gmock/gmock.h"
|
|
|
|
#include "google/protobuf/text_format.h"
|
|
|
|
#include "gtest/gtest.h"
|
|
|
|
|
|
|
|
#include "test/core/util/test_config.h"
|
|
|
|
|
|
|
|
namespace grpc {
|
|
|
|
namespace internal {
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
TEST(GcpObservabilityLoggingSinkTest, LoggingConfigEmpty) {
|
|
|
|
const char* json_str = R"json({
|
|
|
|
"cloud_logging": {
|
|
|
|
}
|
|
|
|
})json";
|
|
|
|
auto json = grpc_core::Json::Parse(json_str);
|
|
|
|
ASSERT_TRUE(json.ok()) << json.status();
|
|
|
|
grpc_core::ValidationErrors errors;
|
|
|
|
auto config = grpc_core::LoadFromJson<GcpObservabilityConfig>(
|
|
|
|
*json, grpc_core::JsonArgs(), &errors);
|
|
|
|
ASSERT_TRUE(errors.ok()) << errors.status("unexpected errors");
|
|
|
|
ObservabilityLoggingSink sink(config.cloud_logging.value(), "test", {});
|
|
|
|
// client test
|
|
|
|
EXPECT_FALSE(sink.FindMatch(true, "foo", "bar").ShouldLog());
|
|
|
|
// server test
|
|
|
|
EXPECT_FALSE(sink.FindMatch(false, "foo", "bar").ShouldLog());
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(GcpObservabilityLoggingSinkTest, LoggingConfigClientWildCardEntries) {
|
|
|
|
const char* json_str = R"json({
|
|
|
|
"cloud_logging": {
|
|
|
|
"client_rpc_events": [
|
|
|
|
{
|
|
|
|
"methods": ["*"],
|
|
|
|
"max_metadata_bytes": 1024,
|
|
|
|
"max_message_bytes": 4096
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
})json";
|
|
|
|
auto json = grpc_core::Json::Parse(json_str);
|
|
|
|
ASSERT_TRUE(json.ok()) << json.status();
|
|
|
|
grpc_core::ValidationErrors errors;
|
|
|
|
auto config = grpc_core::LoadFromJson<GcpObservabilityConfig>(
|
|
|
|
*json, grpc_core::JsonArgs(), &errors);
|
|
|
|
ASSERT_TRUE(errors.ok()) << errors.status("unexpected errors");
|
|
|
|
ObservabilityLoggingSink sink(config.cloud_logging.value(), "test", {});
|
|
|
|
// client test
|
|
|
|
EXPECT_EQ(sink.FindMatch(true, "foo", "bar"),
|
|
|
|
LoggingSink::Config(1024, 4096));
|
|
|
|
// server test
|
|
|
|
EXPECT_FALSE(sink.FindMatch(false, "foo", "bar").ShouldLog());
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(GcpObservabilityLoggingSinkTest, LoggingConfigBadPath) {
|
|
|
|
const char* json_str = R"json({
|
|
|
|
"cloud_logging": {
|
|
|
|
"client_rpc_events": [
|
|
|
|
{
|
|
|
|
"methods": ["*"],
|
|
|
|
"max_metadata_bytes": 1024,
|
|
|
|
"max_message_bytes": 4096
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
})json";
|
|
|
|
auto json = grpc_core::Json::Parse(json_str);
|
|
|
|
ASSERT_TRUE(json.ok()) << json.status();
|
|
|
|
grpc_core::ValidationErrors errors;
|
|
|
|
auto config = grpc_core::LoadFromJson<GcpObservabilityConfig>(
|
|
|
|
*json, grpc_core::JsonArgs(), &errors);
|
|
|
|
ASSERT_TRUE(errors.ok()) << errors.status("unexpected errors");
|
|
|
|
ObservabilityLoggingSink sink(config.cloud_logging.value(), "test", {});
|
|
|
|
EXPECT_FALSE(sink.FindMatch(true, "foo", "").ShouldLog());
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(GcpObservabilityLoggingSinkTest,
|
|
|
|
LoggingConfigClientWildCardServiceEntries) {
|
|
|
|
const char* json_str = R"json({
|
|
|
|
"cloud_logging": {
|
|
|
|
"client_rpc_events": [
|
|
|
|
{
|
|
|
|
"methods": ["service/*"],
|
|
|
|
"max_metadata_bytes": 1024,
|
|
|
|
"max_message_bytes": 4096
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
})json";
|
|
|
|
auto json = grpc_core::Json::Parse(json_str);
|
|
|
|
ASSERT_TRUE(json.ok()) << json.status();
|
|
|
|
grpc_core::ValidationErrors errors;
|
|
|
|
auto config = grpc_core::LoadFromJson<GcpObservabilityConfig>(
|
|
|
|
*json, grpc_core::JsonArgs(), &errors);
|
|
|
|
ASSERT_TRUE(errors.ok()) << errors.status("unexpected errors");
|
|
|
|
ObservabilityLoggingSink sink(config.cloud_logging.value(), "test", {});
|
|
|
|
// client test
|
|
|
|
EXPECT_EQ(sink.FindMatch(true, "service", "bar"),
|
|
|
|
LoggingSink::Config(1024, 4096));
|
|
|
|
EXPECT_FALSE(sink.FindMatch(true, "foo", "bar").ShouldLog());
|
|
|
|
// server test
|
|
|
|
EXPECT_FALSE(sink.FindMatch(false, "service", "bar").ShouldLog());
|
|
|
|
EXPECT_FALSE(sink.FindMatch(false, "foo", "bar").ShouldLog());
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(GcpObservabilityLoggingSinkTest,
|
|
|
|
LoggingConfigClientMultipleMethodEntries) {
|
|
|
|
const char* json_str = R"json({
|
|
|
|
"cloud_logging": {
|
|
|
|
"client_rpc_events": [
|
|
|
|
{
|
|
|
|
"methods": ["foo/bar", "foo/baz"],
|
|
|
|
"max_metadata_bytes": 1024,
|
|
|
|
"max_message_bytes": 4096
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
})json";
|
|
|
|
auto json = grpc_core::Json::Parse(json_str);
|
|
|
|
ASSERT_TRUE(json.ok()) << json.status();
|
|
|
|
grpc_core::ValidationErrors errors;
|
|
|
|
auto config = grpc_core::LoadFromJson<GcpObservabilityConfig>(
|
|
|
|
*json, grpc_core::JsonArgs(), &errors);
|
|
|
|
ASSERT_TRUE(errors.ok()) << errors.status("unexpected errors");
|
|
|
|
ObservabilityLoggingSink sink(config.cloud_logging.value(), "test", {});
|
|
|
|
// client test
|
|
|
|
EXPECT_EQ(sink.FindMatch(true, "foo", "bar"),
|
|
|
|
LoggingSink::Config(1024, 4096));
|
|
|
|
EXPECT_EQ(sink.FindMatch(true, "foo", "baz"),
|
|
|
|
LoggingSink::Config(1024, 4096));
|
|
|
|
// server test
|
|
|
|
EXPECT_FALSE(sink.FindMatch(false, "foo", "bar").ShouldLog());
|
|
|
|
EXPECT_FALSE(sink.FindMatch(false, "foo", "baz").ShouldLog());
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(GcpObservabilityLoggingSinkTest, LoggingConfigClientMultipleEventEntries) {
|
|
|
|
const char* json_str = R"json({
|
|
|
|
"cloud_logging": {
|
|
|
|
"client_rpc_events": [
|
|
|
|
{
|
|
|
|
"methods": ["foo/bar"],
|
|
|
|
"max_metadata_bytes": 1024,
|
|
|
|
"max_message_bytes": 4096
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"methods": ["foo/baz"],
|
|
|
|
"max_metadata_bytes": 512,
|
|
|
|
"max_message_bytes": 2048
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
})json";
|
|
|
|
auto json = grpc_core::Json::Parse(json_str);
|
|
|
|
ASSERT_TRUE(json.ok()) << json.status();
|
|
|
|
grpc_core::ValidationErrors errors;
|
|
|
|
auto config = grpc_core::LoadFromJson<GcpObservabilityConfig>(
|
|
|
|
*json, grpc_core::JsonArgs(), &errors);
|
|
|
|
ASSERT_TRUE(errors.ok()) << errors.status("unexpected errors");
|
|
|
|
ObservabilityLoggingSink sink(config.cloud_logging.value(), "test", {});
|
|
|
|
// client test
|
|
|
|
EXPECT_EQ(sink.FindMatch(true, "foo", "bar"),
|
|
|
|
LoggingSink::Config(1024, 4096));
|
|
|
|
EXPECT_EQ(sink.FindMatch(true, "foo", "baz"), LoggingSink::Config(512, 2048));
|
|
|
|
// server test
|
|
|
|
EXPECT_FALSE(sink.FindMatch(false, "foo", "bar").ShouldLog());
|
|
|
|
EXPECT_FALSE(sink.FindMatch(false, "foo", "baz").ShouldLog());
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(GcpObservabilityLoggingSinkTest, LoggingConfigServerWildCardEntries) {
|
|
|
|
const char* json_str = R"json({
|
|
|
|
"cloud_logging": {
|
|
|
|
"server_rpc_events": [
|
|
|
|
{
|
|
|
|
"methods": ["*"],
|
|
|
|
"max_metadata_bytes": 1024,
|
|
|
|
"max_message_bytes": 4096
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
})json";
|
|
|
|
auto json = grpc_core::Json::Parse(json_str);
|
|
|
|
ASSERT_TRUE(json.ok()) << json.status();
|
|
|
|
grpc_core::ValidationErrors errors;
|
|
|
|
auto config = grpc_core::LoadFromJson<GcpObservabilityConfig>(
|
|
|
|
*json, grpc_core::JsonArgs(), &errors);
|
|
|
|
ASSERT_TRUE(errors.ok()) << errors.status("unexpected errors");
|
|
|
|
ObservabilityLoggingSink sink(config.cloud_logging.value(), "test", {});
|
|
|
|
// client test
|
|
|
|
EXPECT_FALSE(sink.FindMatch(true, "foo", "bar").ShouldLog());
|
|
|
|
// server test
|
|
|
|
EXPECT_EQ(sink.FindMatch(false, "foo", "bar"),
|
|
|
|
LoggingSink::Config(1024, 4096));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(GcpObservabilityLoggingSinkTest,
|
|
|
|
LoggingConfigServerWildCardServiceEntries) {
|
|
|
|
const char* json_str = R"json({
|
|
|
|
"cloud_logging": {
|
|
|
|
"server_rpc_events": [
|
|
|
|
{
|
|
|
|
"methods": ["service/*"],
|
|
|
|
"max_metadata_bytes": 1024,
|
|
|
|
"max_message_bytes": 4096
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
})json";
|
|
|
|
auto json = grpc_core::Json::Parse(json_str);
|
|
|
|
ASSERT_TRUE(json.ok()) << json.status();
|
|
|
|
grpc_core::ValidationErrors errors;
|
|
|
|
auto config = grpc_core::LoadFromJson<GcpObservabilityConfig>(
|
|
|
|
*json, grpc_core::JsonArgs(), &errors);
|
|
|
|
ASSERT_TRUE(errors.ok()) << errors.status("unexpected errors");
|
|
|
|
ObservabilityLoggingSink sink(config.cloud_logging.value(), "test", {});
|
|
|
|
// client test
|
|
|
|
EXPECT_FALSE(sink.FindMatch(true, "service", "bar").ShouldLog());
|
|
|
|
EXPECT_FALSE(sink.FindMatch(true, "foo", "bar").ShouldLog());
|
|
|
|
// server test
|
|
|
|
EXPECT_EQ(sink.FindMatch(false, "service", "bar"),
|
|
|
|
LoggingSink::Config(1024, 4096));
|
|
|
|
EXPECT_FALSE(sink.FindMatch(false, "foo", "bar").ShouldLog());
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(GcpObservabilityLoggingSinkTest,
|
|
|
|
LoggingConfigServerMultipleMethodEntries) {
|
|
|
|
const char* json_str = R"json({
|
|
|
|
"cloud_logging": {
|
|
|
|
"server_rpc_events": [
|
|
|
|
{
|
|
|
|
"methods": ["foo/bar", "foo/baz"],
|
|
|
|
"max_metadata_bytes": 1024,
|
|
|
|
"max_message_bytes": 4096
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
})json";
|
|
|
|
auto json = grpc_core::Json::Parse(json_str);
|
|
|
|
ASSERT_TRUE(json.ok()) << json.status();
|
|
|
|
grpc_core::ValidationErrors errors;
|
|
|
|
auto config = grpc_core::LoadFromJson<GcpObservabilityConfig>(
|
|
|
|
*json, grpc_core::JsonArgs(), &errors);
|
|
|
|
ASSERT_TRUE(errors.ok()) << errors.status("unexpected errors");
|
|
|
|
ObservabilityLoggingSink sink(config.cloud_logging.value(), "test", {});
|
|
|
|
// client test
|
|
|
|
EXPECT_FALSE(sink.FindMatch(true, "foo", "bar").ShouldLog());
|
|
|
|
EXPECT_FALSE(sink.FindMatch(true, "foo", "baz").ShouldLog());
|
|
|
|
// server test
|
|
|
|
EXPECT_EQ(sink.FindMatch(false, "foo", "bar"),
|
|
|
|
LoggingSink::Config(1024, 4096));
|
|
|
|
EXPECT_EQ(sink.FindMatch(false, "foo", "baz"),
|
|
|
|
LoggingSink::Config(1024, 4096));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(GcpObservabilityLoggingSinkTest, LoggingConfigServerMultipleEventEntries) {
|
|
|
|
const char* json_str = R"json({
|
|
|
|
"cloud_logging": {
|
|
|
|
"server_rpc_events": [
|
|
|
|
{
|
|
|
|
"methods": ["foo/bar"],
|
|
|
|
"max_metadata_bytes": 1024,
|
|
|
|
"max_message_bytes": 4096
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"methods": ["foo/baz"],
|
|
|
|
"max_metadata_bytes": 512,
|
|
|
|
"max_message_bytes": 2048
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
})json";
|
|
|
|
auto json = grpc_core::Json::Parse(json_str);
|
|
|
|
ASSERT_TRUE(json.ok()) << json.status();
|
|
|
|
grpc_core::ValidationErrors errors;
|
|
|
|
auto config = grpc_core::LoadFromJson<GcpObservabilityConfig>(
|
|
|
|
*json, grpc_core::JsonArgs(), &errors);
|
|
|
|
ASSERT_TRUE(errors.ok()) << errors.status("unexpected errors");
|
|
|
|
ObservabilityLoggingSink sink(config.cloud_logging.value(), "test", {});
|
|
|
|
// client test
|
|
|
|
EXPECT_FALSE(sink.FindMatch(true, "foo", "bar").ShouldLog());
|
|
|
|
EXPECT_FALSE(sink.FindMatch(true, "foo", "baz").ShouldLog());
|
|
|
|
// server test
|
|
|
|
EXPECT_EQ(sink.FindMatch(false, "foo", "bar"),
|
|
|
|
LoggingSink::Config(1024, 4096));
|
|
|
|
EXPECT_EQ(sink.FindMatch(false, "foo", "baz"),
|
|
|
|
LoggingSink::Config(512, 2048));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(EntryToJsonStructTest, ClientHeader) {
|
|
|
|
LoggingSink::Entry entry;
|
|
|
|
entry.call_id = 1234;
|
|
|
|
entry.sequence_id = 1;
|
|
|
|
entry.type = LoggingSink::Entry::EventType::kClientHeader;
|
|
|
|
entry.payload.metadata["key"] = "value";
|
|
|
|
entry.payload.timeout = grpc_core::Duration::Seconds(100);
|
|
|
|
entry.payload_truncated = true;
|
|
|
|
entry.peer.type = LoggingSink::Entry::Address::Type::kIpv4;
|
|
|
|
entry.peer.address = "127.0.0.1";
|
|
|
|
entry.peer.ip_port = 12345;
|
|
|
|
entry.authority = "authority";
|
|
|
|
entry.service_name = "service_name";
|
|
|
|
entry.method_name = "method_name";
|
|
|
|
|
|
|
|
google::protobuf::Struct proto;
|
|
|
|
EntryToJsonStructProto(std::move(entry), &proto);
|
|
|
|
std::string output;
|
|
|
|
::google::protobuf::TextFormat::PrintToString(proto, &output);
|
|
|
|
const char* pb_str =
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"authority\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"authority\"\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n"
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"callId\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"1234\"\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n"
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"logger\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"LOGGER_UNKNOWN\"\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n"
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"methodName\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"method_name\"\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n"
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"payload\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" struct_value {\n"
|
|
|
|
" fields {\n"
|
|
|
|
" key: \"metadata\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" struct_value {\n"
|
|
|
|
" fields {\n"
|
|
|
|
" key: \"key\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"value\"\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
" fields {\n"
|
|
|
|
" key: \"timeout\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"100.000000000s\"\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n"
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"payloadTruncated\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" bool_value: true\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n"
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"peer\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" struct_value {\n"
|
|
|
|
" fields {\n"
|
|
|
|
" key: \"address\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"127.0.0.1\"\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
" fields {\n"
|
|
|
|
" key: \"ipPort\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" number_value: 12345\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
" fields {\n"
|
|
|
|
" key: \"type\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"TYPE_IPV4\"\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n"
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"sequenceId\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" number_value: 1\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n"
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"serviceName\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"service_name\"\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n"
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"type\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"CLIENT_HEADER\"\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n";
|
|
|
|
EXPECT_EQ(output, pb_str);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(EntryToJsonStructTest, ServerHeader) {
|
|
|
|
LoggingSink::Entry entry;
|
|
|
|
entry.call_id = 1234;
|
|
|
|
entry.sequence_id = 2;
|
|
|
|
entry.type = LoggingSink::Entry::EventType::kServerHeader;
|
|
|
|
entry.logger = LoggingSink::Entry::Logger::kServer;
|
|
|
|
entry.payload.metadata["key"] = "value";
|
|
|
|
entry.peer.type = LoggingSink::Entry::Address::Type::kIpv4;
|
|
|
|
entry.peer.address = "127.0.0.1";
|
|
|
|
entry.peer.ip_port = 1234;
|
|
|
|
entry.authority = "authority";
|
|
|
|
entry.service_name = "service_name";
|
|
|
|
entry.method_name = "method_name";
|
|
|
|
|
|
|
|
google::protobuf::Struct proto;
|
|
|
|
EntryToJsonStructProto(std::move(entry), &proto);
|
|
|
|
std::string output;
|
|
|
|
::google::protobuf::TextFormat::PrintToString(proto, &output);
|
|
|
|
const char* pb_str =
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"authority\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"authority\"\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n"
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"callId\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"1234\"\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n"
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"logger\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"SERVER\"\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n"
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"methodName\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"method_name\"\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n"
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"payload\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" struct_value {\n"
|
|
|
|
" fields {\n"
|
|
|
|
" key: \"metadata\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" struct_value {\n"
|
|
|
|
" fields {\n"
|
|
|
|
" key: \"key\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"value\"\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n"
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"peer\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" struct_value {\n"
|
|
|
|
" fields {\n"
|
|
|
|
" key: \"address\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"127.0.0.1\"\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
" fields {\n"
|
|
|
|
" key: \"ipPort\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" number_value: 1234\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
" fields {\n"
|
|
|
|
" key: \"type\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"TYPE_IPV4\"\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n"
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"sequenceId\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" number_value: 2\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n"
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"serviceName\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"service_name\"\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n"
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"type\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"SERVER_HEADER\"\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n";
|
|
|
|
EXPECT_EQ(output, pb_str);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(EntryToJsonStructTest, ClientMessage) {
|
|
|
|
LoggingSink::Entry entry;
|
|
|
|
entry.call_id = 1234;
|
|
|
|
entry.sequence_id = 3;
|
|
|
|
entry.type = LoggingSink::Entry::EventType::kClientMessage;
|
|
|
|
entry.logger = LoggingSink::Entry::Logger::kClient;
|
|
|
|
entry.payload.message = "hello";
|
|
|
|
entry.payload.message_length = 5;
|
|
|
|
entry.peer.type = LoggingSink::Entry::Address::Type::kIpv4;
|
|
|
|
entry.peer.address = "127.0.0.1";
|
|
|
|
entry.peer.ip_port = 1234;
|
|
|
|
entry.authority = "authority";
|
|
|
|
entry.service_name = "service_name";
|
|
|
|
entry.method_name = "method_name";
|
|
|
|
|
|
|
|
google::protobuf::Struct proto;
|
|
|
|
EntryToJsonStructProto(std::move(entry), &proto);
|
|
|
|
std::string output;
|
|
|
|
::google::protobuf::TextFormat::PrintToString(proto, &output);
|
|
|
|
std::string pb_str = absl::StrFormat(
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"authority\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"authority\"\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n"
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"callId\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"1234\"\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n"
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"logger\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"CLIENT\"\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n"
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"methodName\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"method_name\"\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n"
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"payload\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" struct_value {\n"
|
|
|
|
" fields {\n"
|
|
|
|
" key: \"message\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"%s\"\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
" fields {\n"
|
|
|
|
" key: \"messageLength\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" number_value: 5\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n"
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"peer\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" struct_value {\n"
|
|
|
|
" fields {\n"
|
|
|
|
" key: \"address\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"127.0.0.1\"\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
" fields {\n"
|
|
|
|
" key: \"ipPort\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" number_value: 1234\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
" fields {\n"
|
|
|
|
" key: \"type\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"TYPE_IPV4\"\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n"
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"sequenceId\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" number_value: 3\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n"
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"serviceName\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"service_name\"\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n"
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"type\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"CLIENT_MESSAGE\"\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n",
|
|
|
|
absl::Base64Escape("hello"));
|
|
|
|
EXPECT_EQ(output, pb_str);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(EntryToJsonStructTest, ServerMessage) {
|
|
|
|
LoggingSink::Entry entry;
|
|
|
|
entry.call_id = 1234;
|
|
|
|
entry.sequence_id = 4;
|
|
|
|
entry.type = LoggingSink::Entry::EventType::kServerMessage;
|
|
|
|
entry.logger = LoggingSink::Entry::Logger::kServer;
|
|
|
|
entry.payload.message = "world";
|
|
|
|
entry.payload.message_length = 5;
|
|
|
|
entry.peer.type = LoggingSink::Entry::Address::Type::kIpv4;
|
|
|
|
entry.peer.address = "127.0.0.1";
|
|
|
|
entry.peer.ip_port = 12345;
|
|
|
|
entry.authority = "authority";
|
|
|
|
entry.service_name = "service_name";
|
|
|
|
entry.method_name = "method_name";
|
|
|
|
|
|
|
|
google::protobuf::Struct proto;
|
|
|
|
EntryToJsonStructProto(std::move(entry), &proto);
|
|
|
|
std::string output;
|
|
|
|
::google::protobuf::TextFormat::PrintToString(proto, &output);
|
|
|
|
std::string pb_str = absl::StrFormat(
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"authority\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"authority\"\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n"
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"callId\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"1234\"\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n"
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"logger\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"SERVER\"\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n"
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"methodName\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"method_name\"\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n"
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"payload\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" struct_value {\n"
|
|
|
|
" fields {\n"
|
|
|
|
" key: \"message\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"%s\"\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
" fields {\n"
|
|
|
|
" key: \"messageLength\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" number_value: 5\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n"
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"peer\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" struct_value {\n"
|
|
|
|
" fields {\n"
|
|
|
|
" key: \"address\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"127.0.0.1\"\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
" fields {\n"
|
|
|
|
" key: \"ipPort\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" number_value: 12345\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
" fields {\n"
|
|
|
|
" key: \"type\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"TYPE_IPV4\"\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n"
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"sequenceId\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" number_value: 4\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n"
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"serviceName\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"service_name\"\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n"
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"type\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"SERVER_MESSAGE\"\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n",
|
|
|
|
absl::Base64Escape("world"));
|
|
|
|
EXPECT_EQ(output, pb_str);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(EntryToJsonStructTest, ClientHalfClose) {
|
|
|
|
LoggingSink::Entry entry;
|
|
|
|
entry.call_id = 1234;
|
|
|
|
entry.sequence_id = 5;
|
|
|
|
entry.type = LoggingSink::Entry::EventType::kClientHalfClose;
|
|
|
|
entry.logger = LoggingSink::Entry::Logger::kClient;
|
|
|
|
entry.peer.type = LoggingSink::Entry::Address::Type::kIpv4;
|
|
|
|
entry.peer.address = "127.0.0.1";
|
|
|
|
entry.peer.ip_port = 1234;
|
|
|
|
entry.authority = "authority";
|
|
|
|
entry.service_name = "service_name";
|
|
|
|
entry.method_name = "method_name";
|
|
|
|
|
|
|
|
google::protobuf::Struct proto;
|
|
|
|
EntryToJsonStructProto(std::move(entry), &proto);
|
|
|
|
std::string output;
|
|
|
|
::google::protobuf::TextFormat::PrintToString(proto, &output);
|
|
|
|
const char* pb_str =
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"authority\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"authority\"\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n"
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"callId\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"1234\"\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n"
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"logger\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"CLIENT\"\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n"
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"methodName\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"method_name\"\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n"
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"payload\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" struct_value {\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n"
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"peer\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" struct_value {\n"
|
|
|
|
" fields {\n"
|
|
|
|
" key: \"address\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"127.0.0.1\"\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
" fields {\n"
|
|
|
|
" key: \"ipPort\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" number_value: 1234\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
" fields {\n"
|
|
|
|
" key: \"type\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"TYPE_IPV4\"\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n"
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"sequenceId\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" number_value: 5\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n"
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"serviceName\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"service_name\"\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n"
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"type\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"CLIENT_HALF_CLOSE\"\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n";
|
|
|
|
EXPECT_EQ(output, pb_str);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(EntryToJsonStructTest, ServerTrailer) {
|
|
|
|
LoggingSink::Entry entry;
|
|
|
|
entry.call_id = 1234;
|
|
|
|
entry.sequence_id = 6;
|
|
|
|
entry.type = LoggingSink::Entry::EventType::kServerTrailer;
|
|
|
|
entry.logger = LoggingSink::Entry::Logger::kServer;
|
|
|
|
entry.payload.metadata["key"] = "value";
|
|
|
|
entry.peer.type = LoggingSink::Entry::Address::Type::kIpv4;
|
|
|
|
entry.peer.address = "127.0.0.1";
|
|
|
|
entry.peer.ip_port = 1234;
|
|
|
|
entry.authority = "authority";
|
|
|
|
entry.service_name = "service_name";
|
|
|
|
entry.method_name = "method_name";
|
|
|
|
|
|
|
|
google::protobuf::Struct proto;
|
|
|
|
EntryToJsonStructProto(std::move(entry), &proto);
|
|
|
|
std::string output;
|
|
|
|
::google::protobuf::TextFormat::PrintToString(proto, &output);
|
|
|
|
const char* pb_str =
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"authority\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"authority\"\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n"
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"callId\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"1234\"\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n"
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"logger\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"SERVER\"\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n"
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"methodName\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"method_name\"\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n"
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"payload\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" struct_value {\n"
|
|
|
|
" fields {\n"
|
|
|
|
" key: \"metadata\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" struct_value {\n"
|
|
|
|
" fields {\n"
|
|
|
|
" key: \"key\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"value\"\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n"
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"peer\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" struct_value {\n"
|
|
|
|
" fields {\n"
|
|
|
|
" key: \"address\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"127.0.0.1\"\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
" fields {\n"
|
|
|
|
" key: \"ipPort\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" number_value: 1234\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
" fields {\n"
|
|
|
|
" key: \"type\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"TYPE_IPV4\"\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n"
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"sequenceId\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" number_value: 6\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n"
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"serviceName\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"service_name\"\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n"
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"type\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"SERVER_TRAILER\"\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n";
|
|
|
|
EXPECT_EQ(output, pb_str);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(EntryToJsonStructTest, Cancel) {
|
|
|
|
LoggingSink::Entry entry;
|
|
|
|
entry.call_id = 1234;
|
|
|
|
entry.sequence_id = 7;
|
|
|
|
entry.type = LoggingSink::Entry::EventType::kCancel;
|
|
|
|
entry.logger = LoggingSink::Entry::Logger::kClient;
|
|
|
|
entry.peer.type = LoggingSink::Entry::Address::Type::kIpv4;
|
|
|
|
entry.peer.address = "127.0.0.1";
|
|
|
|
entry.peer.ip_port = 1234;
|
|
|
|
entry.authority = "authority";
|
|
|
|
entry.service_name = "service_name";
|
|
|
|
entry.method_name = "method_name";
|
|
|
|
|
|
|
|
google::protobuf::Struct proto;
|
|
|
|
EntryToJsonStructProto(std::move(entry), &proto);
|
|
|
|
std::string output;
|
|
|
|
::google::protobuf::TextFormat::PrintToString(proto, &output);
|
|
|
|
const char* pb_str =
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"authority\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"authority\"\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n"
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"callId\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"1234\"\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n"
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"logger\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"CLIENT\"\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n"
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"methodName\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"method_name\"\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n"
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"payload\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" struct_value {\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n"
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"peer\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" struct_value {\n"
|
|
|
|
" fields {\n"
|
|
|
|
" key: \"address\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"127.0.0.1\"\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
" fields {\n"
|
|
|
|
" key: \"ipPort\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" number_value: 1234\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
" fields {\n"
|
|
|
|
" key: \"type\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"TYPE_IPV4\"\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n"
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"sequenceId\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" number_value: 7\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n"
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"serviceName\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"service_name\"\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n"
|
|
|
|
"fields {\n"
|
|
|
|
" key: \"type\"\n"
|
|
|
|
" value {\n"
|
|
|
|
" string_value: \"CANCEL\"\n"
|
|
|
|
" }\n"
|
|
|
|
"}\n";
|
|
|
|
EXPECT_EQ(output, pb_str);
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
} // namespace internal
|
|
|
|
} // namespace grpc
|
|
|
|
|
|
|
|
int main(int argc, char** argv) {
|
|
|
|
grpc::testing::TestEnvironment env(&argc, argv);
|
|
|
|
::testing::InitGoogleTest(&argc, argv);
|
|
|
|
return RUN_ALL_TESTS();
|
|
|
|
}
|