userver: /data/code/userver/libraries/protobuf/tests/json/field_mask_to_json_test.cpp Source File
Loading...
Searching...
No Matches
field_mask_to_json_test.cpp
1#include <gtest/gtest.h>
2
3#include <algorithm>
4#include <memory>
5#include <ostream>
6#include <string>
7#include <vector>
8
9#include <fmt/format.h>
10#include <fmt/ranges.h>
11#include <google/protobuf/dynamic_message.h>
12
13#include <userver/protobuf/json/convert.hpp>
14#include <userver/utest/assert_macros.hpp>
15
16#include "utils.hpp"
17
18USERVER_NAMESPACE_BEGIN
19
20namespace protobuf::json::tests {
21
22struct FieldMaskToJsonSuccessTestParam {
23 FieldMaskMessageData input = {};
24 std::string expected_json = {};
25 PrintOptions options = {};
26};
27
28struct FieldMaskToJsonFailureTestParam {
29 FieldMaskMessageData input = {};
30 PrintErrorCode expected_errc = {};
31 std::string expected_path = {};
32 PrintOptions options = {};
33};
34
35std::vector<std::string> ParseFieldMaskStr(std::string_view paths) {
36 std::vector<std::string> result;
37 std::size_t pos = 0;
38
39 while (true) {
40 auto comma_pos = paths.find(',', pos);
41
42 if (comma_pos == std::string_view::npos) {
43 result.emplace_back(paths.substr(pos));
44 break;
45 }
46
47 result.emplace_back(paths.substr(pos, comma_pos - pos));
48 pos = comma_pos + 1;
49 }
50
51 return result;
52}
53
54void PrintTo(const FieldMaskToJsonSuccessTestParam& param, std::ostream* os) {
55 if (param.input.field1) {
56 *os << fmt::format("{{ input = {{.field1='{}'}} }}", param.input.field1.value());
57 } else {
58 *os << fmt::format("{{ input = {{.field1=nullopt}} }}");
59 }
60}
61
62void PrintTo(const FieldMaskToJsonFailureTestParam& param, std::ostream* os) {
63 if (param.input.field1) {
64 *os << fmt::format("{{ input = {{.field1='{}'}} }}", param.input.field1.value());
65 } else {
66 *os << fmt::format("{{ input = {{.field1=nullopt}} }}");
67 }
68}
69
70class FieldMaskToJsonSuccessTest : public ::testing::TestWithParam<FieldMaskToJsonSuccessTestParam> {};
71class FieldMaskToJsonFailureTest : public ::testing::TestWithParam<FieldMaskToJsonFailureTestParam> {};
72
73INSTANTIATE_TEST_SUITE_P(
74 ,
75 FieldMaskToJsonSuccessTest,
76 ::testing::Values(
77 FieldMaskToJsonSuccessTestParam{FieldMaskMessageData{}, R"({})"},
78 FieldMaskToJsonSuccessTestParam{FieldMaskMessageData{std::vector<std::string>{}}, R"({"field1":""})"},
79 FieldMaskToJsonSuccessTestParam{FieldMaskMessageData{std::vector<std::string>{""}}, R"({"field1":""})"},
80 FieldMaskToJsonSuccessTestParam{
81 FieldMaskMessageData{std::vector<std::string>{"", "", "", "aaa", ""}},
82 R"({"field1":",,,aaa,"})"
83 },
84 FieldMaskToJsonSuccessTestParam{
85 FieldMaskMessageData{std::vector<std::string>{"some_field"}},
86 R"({"field1":"someField"})",
87 {.preserve_proto_field_names = true} // does not affect field mask serialization!
88 },
89 FieldMaskToJsonSuccessTestParam{
90 FieldMaskMessageData{std::vector<std::string>{"some_field.another_field..one_more", "_a_b0_c"}},
91 R"({"field1":"someField.anotherField..oneMore,AB0C"})"
92 }
93 )
94);
95
96// Protobuf ProtoJSON in legacy mode does not return error on invalid characters, which we
97// want to prohibit (because we do not want our clients to use syntax that may break in the
98// newer protobuf versions).
99
100INSTANTIATE_TEST_SUITE_P(
101 ,
102 FieldMaskToJsonFailureTest,
103 ::testing::Values(
104 FieldMaskToJsonFailureTestParam{
105 FieldMaskMessageData{std::vector<std::string>{"Some_field"}},
106 PrintErrorCode::kInvalidValue,
107 "field1"
108 },
109 FieldMaskToJsonFailureTestParam{
110 FieldMaskMessageData{std::vector<std::string>{"some_Field"}},
111 PrintErrorCode::kInvalidValue,
112 "field1"
113 },
114 FieldMaskToJsonFailureTestParam{
115 FieldMaskMessageData{std::vector<std::string>{"some_fielD"}},
116 PrintErrorCode::kInvalidValue,
117 "field1"
118 },
119 FieldMaskToJsonFailureTestParam{
120 FieldMaskMessageData{std::vector<std::string>{"some_f!ield"}},
121 PrintErrorCode::kInvalidValue,
122 "field1"
123 },
124 FieldMaskToJsonFailureTestParam{
125 FieldMaskMessageData{std::vector<std::string>{"__some_field"}},
126 PrintErrorCode::kInvalidValue,
127 "field1"
128 },
129 FieldMaskToJsonFailureTestParam{
130 FieldMaskMessageData{std::vector<std::string>{"some__field"}},
131 PrintErrorCode::kInvalidValue,
132 "field1"
133 },
134 FieldMaskToJsonFailureTestParam{
135 FieldMaskMessageData{std::vector<std::string>{"some_field_"}},
136 PrintErrorCode::kInvalidValue,
137 "field1"
138 },
139 FieldMaskToJsonFailureTestParam{
140 FieldMaskMessageData{std::vector<std::string>{"some_0field"}},
141 PrintErrorCode::kInvalidValue,
142 "field1"
143 }
144 )
145);
146
147TEST_P(FieldMaskToJsonSuccessTest, Test) {
148 const auto& param = GetParam();
149
150 auto input = PrepareTestData(param.input);
151 formats::json::Value json, expected_json, sample_json;
152
153 UASSERT_NO_THROW((json = MessageToJson(input, param.options)));
154 UASSERT_NO_THROW((expected_json = PrepareJsonTestData(param.expected_json)));
155 UASSERT_NO_THROW((sample_json = CreateSampleJson(input, param.options)));
156
157 if (!json.HasMember("field1")) {
158 EXPECT_FALSE(expected_json.HasMember("field1"));
159 EXPECT_FALSE(sample_json.HasMember("field1"));
160 return;
161 }
162
163 ASSERT_TRUE(json["field1"].IsString());
164 ASSERT_TRUE(expected_json["field1"].IsString());
165 ASSERT_TRUE(sample_json["field1"].IsString());
166
167 auto json_paths = ParseFieldMaskStr(json["field1"].As<std::string>());
168 auto expected_json_paths = ParseFieldMaskStr(expected_json["field1"].As<std::string>());
169 auto sample_json_paths = ParseFieldMaskStr(sample_json["field1"].As<std::string>());
170
171 std::sort(json_paths.begin(), json_paths.end());
172 std::sort(expected_json_paths.begin(), expected_json_paths.end());
173 std::sort(sample_json_paths.begin(), sample_json_paths.end());
174
175 EXPECT_EQ(json_paths, expected_json_paths);
176 EXPECT_EQ(expected_json_paths, sample_json_paths);
177}
178
179TEST_P(FieldMaskToJsonFailureTest, Test) {
180 const auto& param = GetParam();
181 auto input = PrepareTestData(param.input);
182
183 EXPECT_PRINT_ERROR((void)MessageToJson(input, param.options), param.expected_errc, param.expected_path);
184}
185
186TEST(FieldMaskToJsonAdditionalTest, InlinedNonNull) {
187 FieldMaskMessageData data{std::vector<std::string>{"some_field.nested_field"}};
188 auto message = PrepareTestData(data);
189 formats::json::Value json, sample;
190
191 UASSERT_NO_THROW((json = MessageToJson(message.field1(), {})));
192 UASSERT_NO_THROW((sample = CreateSampleJson(message.field1())));
193 ASSERT_TRUE(json.IsString());
194 ASSERT_TRUE(sample.IsString());
195
196 auto paths = ParseFieldMaskStr(json.As<std::string>());
197 auto sample_paths = ParseFieldMaskStr(sample.As<std::string>());
198
199 std::sort(paths.begin(), paths.end());
200 std::sort(sample_paths.begin(), sample_paths.end());
201
202 EXPECT_EQ(paths, sample_paths);
203 EXPECT_EQ(paths, std::vector<std::string>{"someField.nestedField"});
204}
205
206TEST(FieldMaskToJsonAdditionalTest, InlinedNull) {
208 auto message = PrepareTestData(data);
209 formats::json::Value json, sample;
210
211 UASSERT_NO_THROW((json = MessageToJson(message.field1(), {})));
212 UASSERT_NO_THROW((sample = CreateSampleJson(message.field1())));
213 ASSERT_TRUE(json.IsString());
214 ASSERT_TRUE(sample.IsString());
215 EXPECT_TRUE(json.As<std::string>().empty());
216 EXPECT_TRUE(sample.As<std::string>().empty());
217}
218
219TEST(FieldMaskToJsonAdditionalTest, DynamicMessage) {
220 using Message = ::google::protobuf::FieldMask;
221 ::google::protobuf::DynamicMessageFactory factory;
222
223 {
224 std::unique_ptr<::google::protobuf::Message> message(factory.GetPrototype(Message::descriptor())->New());
225 const auto reflection = message->GetReflection();
226 const auto paths_desc = message->GetDescriptor()->FindFieldByName("paths");
227
228 reflection->AddString(message.get(), paths_desc, "some_field.another_field");
229 reflection->AddString(message.get(), paths_desc, "one_more");
230
231 formats::json::Value json;
232
233 UASSERT_NO_THROW((json = MessageToJson(*message, {})));
234 ASSERT_TRUE(json.IsString());
235
236 auto paths = ParseFieldMaskStr(json.As<std::string>());
237 std::sort(paths.begin(), paths.end());
238
239 ASSERT_EQ(paths.size(), std::size_t{2});
240 EXPECT_EQ(paths[0], "oneMore");
241 EXPECT_EQ(paths[1], "someField.anotherField");
242 }
243}
244
245} // namespace protobuf::json::tests
246
247USERVER_NAMESPACE_END