The C based gRPC (C++, Python, Ruby, Objective-C, PHP, C#) https://grpc.io/
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

1082 lines
31 KiB

//
// 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 "src/core/lib/json/json_reader.h"
#include "test/core/util/test_config.h"
namespace grpc {
namespace internal {
namespace {
using grpc_core::LoggingSink;
TEST(GcpObservabilityLoggingSinkTest, LoggingConfigEmpty) {
const char* json_str = R"json({
"cloud_logging": {
}
})json";
auto json = grpc_core::JsonParse(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(absl::StatusCode::kInvalidArgument,
"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::JsonParse(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(absl::StatusCode::kInvalidArgument,
"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::JsonParse(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(absl::StatusCode::kInvalidArgument,
"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::JsonParse(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(absl::StatusCode::kInvalidArgument,
"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::JsonParse(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(absl::StatusCode::kInvalidArgument,
"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::JsonParse(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(absl::StatusCode::kInvalidArgument,
"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::JsonParse(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(absl::StatusCode::kInvalidArgument,
"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::JsonParse(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(absl::StatusCode::kInvalidArgument,
"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::JsonParse(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(absl::StatusCode::kInvalidArgument,
"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::JsonParse(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(absl::StatusCode::kInvalidArgument,
"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: \"00000000-0000-4000-8000-0000000004d2\"\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: \"00000000-0000-4000-8000-0000000004d2\"\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: \"00000000-0000-4000-8000-0000000004d2\"\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: \"00000000-0000-4000-8000-0000000004d2\"\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: \"00000000-0000-4000-8000-0000000004d2\"\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: \"00000000-0000-4000-8000-0000000004d2\"\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: \"00000000-0000-4000-8000-0000000004d2\"\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();
}