Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add support for proto ListValue formatting #14517

Closed
wants to merge 1 commit into from
Closed
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
178 changes: 122 additions & 56 deletions source/common/formatter/substitution_formatter.cc
Original file line number Diff line number Diff line change
Expand Up @@ -49,9 +49,6 @@ const std::regex& getStartTimeNewlinePattern() {
}
const std::regex& getNewlinePattern() { CONSTRUCT_ON_FIRST_USE(std::regex, "\n"); }

template <class... Ts> struct StructFormatMapVisitor : Ts... { using Ts::operator()...; };
template <class... Ts> StructFormatMapVisitor(Ts...) -> StructFormatMapVisitor<Ts...>;

} // namespace

const std::string SubstitutionFormatUtils::DEFAULT_FORMAT =
Expand Down Expand Up @@ -142,76 +139,145 @@ std::string JsonFormatterImpl::format(const Http::RequestHeaderMap& request_head
return absl::StrCat(log_line, "\n");
}

StructFormatter::StructFormatter(const ProtobufWkt::Struct& format_mapping, bool preserve_types,
bool omit_empty_values)
: omit_empty_values_(omit_empty_values), preserve_types_(preserve_types),
empty_value_(omit_empty_values_ ? EMPTY_STRING : DefaultUnspecifiedValueString),
struct_output_format_(toFormatValue(format_mapping)) {}

StructFormatter::StructFormatMapWrapper
StructFormatter::toFormatMap(const ProtobufWkt::Struct& struct_format) const {
StructFormatter::toFormatValue(const ProtobufWkt::Struct& struct_format) const {
auto output = std::make_unique<StructFormatMap>();
for (const auto& pair : struct_format.fields()) {
switch (pair.second.kind_case()) {
case ProtobufWkt::Value::kStringValue:
output->emplace(pair.first, SubstitutionFormatParser::parse(pair.second.string_value()));
output->emplace(pair.first, toFormatValue(pair.second.string_value()));
break;

case ProtobufWkt::Value::kStructValue:
output->emplace(pair.first, toFormatValue(pair.second.struct_value()));
break;

case ProtobufWkt::Value::kListValue:
output->emplace(pair.first, toFormatValue(pair.second.list_value()));
break;

default:
throw EnvoyException("Only string values, nested structs and list values are "
"supported in structured access log format.");
}
}
return {std::move(output)};
};

StructFormatter::StructFormatListWrapper
StructFormatter::toFormatValue(const ProtobufWkt::ListValue& list_value_format) const {
auto output = std::make_unique<StructFormatList>();
for (const auto& value : list_value_format.values()) {
switch (value.kind_case()) {
case ProtobufWkt::Value::kStringValue:
output->emplace_back(toFormatValue(value.string_value()));
break;

case ProtobufWkt::Value::kStructValue:
output->emplace(pair.first, toFormatMap(pair.second.struct_value()));
output->emplace_back(toFormatValue(value.struct_value()));
break;

case ProtobufWkt::Value::kListValue:
output->emplace_back(toFormatValue(value.list_value()));
break;
default:
throw EnvoyException(
"Only string values or nested structs are supported in structured access log format.");
throw EnvoyException("Only string values, nested structs and list values are "
"supported in structured access log format.");
}
}
return {std::move(output)};
}

std::vector<FormatterProviderPtr>
StructFormatter::toFormatValue(const std::string& string_format) const {
return SubstitutionFormatParser::parse(string_format);
};

ProtobufWkt::Value StructFormatter::providersCallback(
const std::vector<FormatterProviderPtr>& providers,
const Http::RequestHeaderMap& request_headers, const Http::ResponseHeaderMap& response_headers,
const Http::ResponseTrailerMap& response_trailers, const StreamInfo::StreamInfo& stream_info,
absl::string_view local_reply_body) const {
ASSERT(!providers.empty());
if (providers.size() == 1) {
const auto& provider = providers.front();
if (preserve_types_) {
return provider->formatValue(request_headers, response_headers, response_trailers,
stream_info, local_reply_body);
}

if (omit_empty_values_) {
return ValueUtil::optionalStringValue(provider->format(
request_headers, response_headers, response_trailers, stream_info, local_reply_body));
}

const auto str = provider->format(request_headers, response_headers, response_trailers,
stream_info, local_reply_body);
return ValueUtil::stringValue(str.value_or(DefaultUnspecifiedValueString));
}
// Multiple providers forces string output.
std::string str;
for (const auto& provider : providers) {
const auto bit = provider->format(request_headers, response_headers, response_trailers,
stream_info, local_reply_body);
str += bit.value_or(empty_value_);
}
return ValueUtil::stringValue(str);
}

ProtobufWkt::Value StructFormatter::structFormatMapCallback(
const StructFormatter::StructFormatMapWrapper& format_map,
const StructFormatter::StructFormatMapVisitor& visitor) const {
ProtobufWkt::Struct output;
auto* fields = output.mutable_fields();
for (const auto& pair : *format_map.value_) {
ProtobufWkt::Value value = absl::visit(visitor, pair.second);
if (omit_empty_values_ && value.kind_case() == ProtobufWkt::Value::kNullValue) {
continue;
}
(*fields)[pair.first] = value;
}
return ValueUtil::structValue(output);
}

ProtobufWkt::Value StructFormatter::structFormatListCallback(
const StructFormatter::StructFormatListWrapper& format_list,
const StructFormatter::StructFormatMapVisitor& visitor) const {
std::vector<ProtobufWkt::Value> output;
for (const auto& val : *format_list.value_) {
ProtobufWkt::Value value = absl::visit(visitor, val);
if (omit_empty_values_ && value.kind_case() == ProtobufWkt::Value::kNullValue) {
continue;
}
output.push_back(value);
}
return ValueUtil::listValue(output);
}

ProtobufWkt::Struct StructFormatter::format(const Http::RequestHeaderMap& request_headers,
const Http::ResponseHeaderMap& response_headers,
const Http::ResponseTrailerMap& response_trailers,
const StreamInfo::StreamInfo& stream_info,
absl::string_view local_reply_body) const {
const std::string& empty_value =
omit_empty_values_ ? EMPTY_STRING : DefaultUnspecifiedValueString;
const std::function<ProtobufWkt::Value(const std::vector<FormatterProviderPtr>&)>
providers_callback = [&](const std::vector<FormatterProviderPtr>& providers) {
ASSERT(!providers.empty());
if (providers.size() == 1) {
const auto& provider = providers.front();
if (preserve_types_) {
return provider->formatValue(request_headers, response_headers, response_trailers,
stream_info, local_reply_body);
}

if (omit_empty_values_) {
return ValueUtil::optionalStringValue(
provider->format(request_headers, response_headers, response_trailers, stream_info,
local_reply_body));
}

const auto str = provider->format(request_headers, response_headers, response_trailers,
stream_info, local_reply_body);
return ValueUtil::stringValue(str.value_or(DefaultUnspecifiedValueString));
}
// Multiple providers forces string output.
std::string str;
for (const auto& provider : providers) {
const auto bit = provider->format(request_headers, response_headers, response_trailers,
stream_info, local_reply_body);
str += bit.value_or(empty_value);
}
return ValueUtil::stringValue(str);
};
const std::function<ProtobufWkt::Value(const StructFormatter::StructFormatMapWrapper&)>
struct_format_map_callback = [&](const StructFormatter::StructFormatMapWrapper& format) {
ProtobufWkt::Struct output;
auto* fields = output.mutable_fields();
StructFormatMapVisitor visitor{struct_format_map_callback, providers_callback};
for (const auto& pair : *format.value_) {
ProtobufWkt::Value value = absl::visit(visitor, pair.second);
if (omit_empty_values_ && value.kind_case() == ProtobufWkt::Value::kNullValue) {
continue;
}
(*fields)[pair.first] = value;
}
return ValueUtil::structValue(output);
};
return struct_format_map_callback(struct_output_format_).struct_value();
StructFormatMapVisitor visitor{
[&](const std::vector<FormatterProviderPtr>& providers) {
return providersCallback(providers, request_headers, response_headers, response_trailers,
stream_info, local_reply_body);
},
[&, this](const StructFormatter::StructFormatMapWrapper& format_map) {
return structFormatMapCallback(format_map, visitor);
},
[&, this](const StructFormatter::StructFormatListWrapper& format_list) {
return structFormatListCallback(format_list, visitor);
},
};
return structFormatMapCallback(struct_output_format_, visitor).struct_value();
}

void SubstitutionFormatParser::parseCommandHeader(const std::string& token, const size_t start,
Expand Down Expand Up @@ -1027,8 +1093,8 @@ MetadataFormatter::formatMetadataValue(const envoy::config::core::v3::Metadata&
return val;
}

// TODO(glicht): Consider adding support for route/listener/cluster metadata as suggested by @htuch.
// See: https://github.com/envoyproxy/envoy/issues/3006
// TODO(glicht): Consider adding support for route/listener/cluster metadata as suggested by
// @htuch. See: https://github.com/envoyproxy/envoy/issues/3006
DynamicMetadataFormatter::DynamicMetadataFormatter(const std::string& filter_namespace,
const std::vector<std::string>& path,
absl::optional<size_t> max_length)
Expand Down
50 changes: 42 additions & 8 deletions source/common/formatter/substitution_formatter.h
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
#pragma once

#include <functional>
#include <list>
#include <string>
#include <vector>

Expand Down Expand Up @@ -114,9 +115,7 @@ class FormatterImpl : public Formatter {
class StructFormatter {
public:
StructFormatter(const ProtobufWkt::Struct& format_mapping, bool preserve_types,
bool omit_empty_values)
: omit_empty_values_(omit_empty_values), preserve_types_(preserve_types),
struct_output_format_(toFormatMap(format_mapping)) {}
bool omit_empty_values);

ProtobufWkt::Struct format(const Http::RequestHeaderMap& request_headers,
const Http::ResponseHeaderMap& response_headers,
Expand All @@ -126,22 +125,57 @@ class StructFormatter {

private:
struct StructFormatMapWrapper;
using StructFormatMapValue =
absl::variant<const std::vector<FormatterProviderPtr>, const StructFormatMapWrapper>;
struct StructFormatListWrapper;
using StructFormatValue =
absl::variant<const std::vector<FormatterProviderPtr>, const StructFormatMapWrapper,
const StructFormatListWrapper>;
// Although not required for Struct/JSON, it is nice to have the order of
// properties preserved between the format and the log entry, thus std::map.
using StructFormatMap = std::map<std::string, StructFormatMapValue>;
using StructFormatMap = std::map<std::string, StructFormatValue>;
using StructFormatMapPtr = std::unique_ptr<StructFormatMap>;
struct StructFormatMapWrapper {
StructFormatMapPtr value_;
};

StructFormatMapWrapper toFormatMap(const ProtobufWkt::Struct& struct_format) const;
using StructFormatList = std::list<StructFormatValue>;
using StructFormatListPtr = std::unique_ptr<StructFormatList>;
struct StructFormatListWrapper {
StructFormatListPtr value_;
};

template <class... Ts> struct StructFormatMapVisitorHelper : Ts... { using Ts::operator()...; };
template <class... Ts> StructFormatMapVisitorHelper(Ts...) -> StructFormatMapVisitorHelper<Ts...>;
using StructFormatMapVisitor = StructFormatMapVisitorHelper<
const std::function<ProtobufWkt::Value(const std::vector<FormatterProviderPtr>&)>,
const std::function<ProtobufWkt::Value(const StructFormatter::StructFormatMapWrapper&)>,
const std::function<ProtobufWkt::Value(const StructFormatter::StructFormatListWrapper&)>>;

// Methods for building the format map.
std::vector<FormatterProviderPtr> toFormatValue(const std::string& string_format) const;
StructFormatMapWrapper toFormatValue(const ProtobufWkt::Struct& struct_format) const;
StructFormatListWrapper toFormatValue(const ProtobufWkt::ListValue& list_value_format) const;

// Methods for doing the actual formatting.
ProtobufWkt::Value providersCallback(const std::vector<FormatterProviderPtr>& providers,
const Http::RequestHeaderMap& request_headers,
const Http::ResponseHeaderMap& response_headers,
const Http::ResponseTrailerMap& response_trailers,
const StreamInfo::StreamInfo& stream_info,
absl::string_view local_reply_body) const;
ProtobufWkt::Value
structFormatMapCallback(const StructFormatter::StructFormatMapWrapper& format_map,
const StructFormatMapVisitor& visitor) const;
ProtobufWkt::Value
structFormatListCallback(const StructFormatter::StructFormatListWrapper& format_list,
const StructFormatMapVisitor& visitor) const;

const bool omit_empty_values_;
const bool preserve_types_;
const std::string empty_value_;
const StructFormatMapWrapper struct_output_format_;
};
}; // namespace Formatter

using StructFormatterPtr = std::unique_ptr<StructFormatter>;

class JsonFormatterImpl : public Formatter {
public:
Expand Down
3 changes: 2 additions & 1 deletion test/common/formatter/substitution_format_string_test.cc
Original file line number Diff line number Diff line change
Expand Up @@ -93,7 +93,8 @@ TEST_F(SubstitutionFormatStringUtilsTest, TestInvalidConfigs) {
TestUtility::loadFromYaml(yaml, config_);
EXPECT_THROW_WITH_MESSAGE(
SubstitutionFormatStringUtils::fromProtoConfig(config_, context_.api()), EnvoyException,
"Only string values or nested structs are supported in structured access log format.");
"Only string values, nested structs and list values are supported in structured access log "
"format.");
}
}

Expand Down
32 changes: 32 additions & 0 deletions test/common/formatter/substitution_formatter_test.cc
Original file line number Diff line number Diff line change
Expand Up @@ -1613,6 +1613,38 @@ TEST(SubstitutionFormatterTest, StructFormatterNestedObject) {
EXPECT_TRUE(TestUtility::protoEqual(out_struct, expected));
}

TEST(SubstitutionFormatterTest, StructFormatterList) {
StreamInfo::MockStreamInfo stream_info;
Http::TestRequestHeaderMapImpl request_header;
Http::TestResponseHeaderMapImpl response_header;
Http::TestResponseTrailerMapImpl response_trailer;
std::string body;

envoy::config::core::v3::Metadata metadata;
populateMetadataTestData(metadata);
absl::optional<Http::Protocol> protocol = Http::Protocol::Http11;
EXPECT_CALL(stream_info, protocol()).WillRepeatedly(Return(protocol));

ProtobufWkt::Struct key_mapping;
TestUtility::loadFromYaml(R"EOF(
list:
- plain_string: plain_string_value
- protocol: '%PROTOCOL%'
)EOF",
key_mapping);
StructFormatter formatter(key_mapping, false, false);

const ProtobufWkt::Struct expected = TestUtility::jsonToStruct(R"EOF({
"list": [
{ "plain_string": "plain_string_value" },
{ "protocol": "HTTP/1.1" },
]
})EOF");
const ProtobufWkt::Struct out_struct =
formatter.format(request_header, response_header, response_trailer, stream_info, body);
EXPECT_TRUE(TestUtility::protoEqual(out_struct, expected));
}

TEST(SubstitutionFormatterTest, StructFormatterSingleOperatorTest) {
StreamInfo::MockStreamInfo stream_info;
Http::TestRequestHeaderMapImpl request_header;
Expand Down