blob: 3028ac84995491c46574b33910d15d2e70e9922e [file] [log] [blame]
// Copyright 2022 Google LLC
//
// 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.
// Tests of Arbitrary<T> domains.
#include <algorithm>
#include <cstdint>
#include <iostream>
#include <optional>
#include <string>
#include <thread> // NOLINT(build/c++11)
#include <type_traits>
#include <utility>
#include <vector>
#include "gmock/gmock.h"
#include "gtest/gtest.h"
#include "absl/container/flat_hash_set.h"
#include "absl/random/bit_gen_ref.h"
#include "absl/random/random.h"
#include "absl/random/seed_sequences.h"
#include "absl/strings/match.h"
#include "absl/strings/string_view.h"
#include "absl/time/clock.h"
#include "absl/time/time.h"
#include "./fuzztest/domain.h" // IWYU pragma: keep
#include "./domain_tests/domain_testing.h"
#include "./fuzztest/internal/test_protobuf.pb.h"
#include "google/protobuf/descriptor.h"
#include "google/protobuf/message.h"
#include "google/protobuf/message_lite.h"
#include "google/protobuf/util/message_differencer.h"
namespace fuzztest {
namespace {
using ::fuzztest::internal::ProtoExtender;
using ::fuzztest::internal::TestProtobuf;
using ::fuzztest::internal::TestProtobuf_Enum;
using ::fuzztest::internal::TestProtobufWithExtension;
using ::fuzztest::internal::TestProtobufWithRequired;
using ::fuzztest::internal::TestSubProtobuf;
using ::google::protobuf::FieldDescriptor;
using ::testing::Contains;
using ::testing::Each;
using ::testing::ElementsAre;
using ::testing::Ge;
using ::testing::Gt;
using ::testing::IsTrue;
using ::testing::ResultOf;
using ::testing::SizeIs;
using ::testing::UnorderedElementsAre;
TEST(ArbitraryProtocolBufferTest, InitGeneratesSeeds) {
TestProtobuf seed;
seed.set_i32(42);
seed.set_str("Hello");
EXPECT_THAT(
GenerateInitialValues(Arbitrary<TestProtobuf>().WithSeeds({seed}), 1000),
Contains(ResultOf(
[&seed](const auto& val) {
return google::protobuf::util::MessageDifferencer::Equals(val.user_value,
seed);
},
IsTrue())));
}
// TODO(b/246448769): Rewrite the test to decrease the chance of failure.
TEST(ProtocolBuffer,
RepeatedMutationEventuallyMutatesAllFieldsOfArbitraryProtobuf) {
Domain<TestProtobuf> domain = Arbitrary<TestProtobuf>();
absl::BitGen bitgen;
Value val(domain, bitgen);
const auto verify_field_changes = [&](absl::string_view name, auto has,
auto get) {
const auto optional_get = [&]() {
return has(val.user_value) ? std::optional(get(val.user_value))
: std::nullopt;
};
using OptionalV = decltype(optional_get());
Set<OptionalV> values;
int iterations = 10'000;
while (--iterations > 0 && values.size() < 2) {
values.insert(optional_get());
val.Mutate(domain, bitgen, {}, false);
}
EXPECT_GT(iterations, 0)
<< "Field: " << name << " -- " << testing::PrintToString(values);
};
const auto verify_repeated_field_changes = [&](absl::string_view name,
auto get) {
Set<int> sizes;
Set<std::decay_t<decltype(get(val.user_value)[0])>> elem0;
int iterations = 10'000;
while (--iterations > 0 && (elem0.size() < 2 || sizes.size() < 2)) {
auto field = get(val.user_value);
sizes.insert(field.size());
if (field.size() > 0) {
elem0.insert(field[0]);
}
val.Mutate(domain, bitgen, {}, false);
}
EXPECT_GT(iterations, 0)
<< "Field: " << name << " -- " << testing::PrintToString(sizes)
<< " ++ " << testing::PrintToString(elem0);
};
VisitTestProtobuf(verify_field_changes, verify_repeated_field_changes);
VerifyRoundTripThroughConversion(val, domain);
}
TEST(ProtocolBuffer, RepeatedMutationEventuallyMutatesExtensionFields) {
auto has_ext = ResultOf(
[](const auto& val) {
return val.user_value.HasExtension(internal::ProtoExtender::ext);
},
IsTrue());
auto has_rep_ext = ResultOf(
[](const auto& val) {
return val.user_value.ExtensionSize(internal::ProtoExtender::rep_ext);
},
Gt(0));
EXPECT_THAT(
GenerateNonUniqueValues(Arbitrary<TestProtobufWithExtension>(), 1, 5000),
AllOf(Contains(has_ext), Contains(has_rep_ext)));
}
// TODO(b/246652379): Re-enable after b/231212420 is fixed.
TEST(ProtocolBuffer,
DISABLED_ShrinkingEventuallyUnsetsAndEmptiesAllFieldsOfArbitraryProtobuf) {
Domain<TestProtobuf> domain = Arbitrary<TestProtobuf>();
absl::BitGen bitgen;
Value val(domain, bitgen);
for (int i = 0; i < 10'000; ++i) {
val.Mutate(domain, bitgen, {}, false);
}
// We verify that the object actually has things in it. This can technically
// fail if the very last operation done above was to unset the very last set
// field, but it is very unlikely.
ASSERT_NE(val.user_value.ByteSizeLong(), 0);
// ByteSizeLong() == 0 is a simple way to determine that all fields are unset.
for (int iteration = 0;
val.user_value.ByteSizeLong() > 0 && iteration < 50'000; ++iteration) {
const auto prev = val;
val.Mutate(domain, bitgen, {}, true);
ASSERT_TRUE(TowardsZero(prev.user_value, val.user_value))
<< prev << " -vs- " << val;
}
EXPECT_EQ(val.user_value.ByteSizeLong(), 0);
}
TEST(ProtocolBufferWithRequiredFields, OptionalFieldIsEventuallySet) {
auto domain = Arbitrary<TestProtobufWithRequired>()
.WithRepeatedFieldsMaxSize(0)
.WithProtobufFieldUnset("sub_req");
absl::BitGen bitgen;
Value val(domain, bitgen);
ASSERT_TRUE(val.user_value.IsInitialized()) << val.user_value;
for (int i = 0; i < 1000; ++i) {
val.Mutate(domain, bitgen, {}, false);
ASSERT_TRUE(val.user_value.IsInitialized()) << val.user_value;
if (val.user_value.has_i32()) break;
}
EXPECT_TRUE(val.user_value.has_i32());
}
TEST(ProtocolBufferWithRequiredFields, OptionalFieldIsEventuallyUnset) {
auto domain = Arbitrary<TestProtobufWithRequired>()
.WithRepeatedFieldsMaxSize(0)
.WithProtobufFieldUnset("sub_req");
absl::BitGen bitgen;
Value val(domain, bitgen);
ASSERT_TRUE(val.user_value.IsInitialized()) << val.user_value;
// With the restricted domain, the probability of unsetting the field i32 is
// at least 1/800. Hence, within 11000 iterations we'll fail to observe this
// event with probability at most 10^(-6).
for (int i = 0; i < 11000; ++i) {
val.Mutate(domain, bitgen, {}, false);
ASSERT_TRUE(val.user_value.IsInitialized()) << val.user_value;
if (!val.user_value.has_i32()) break;
}
EXPECT_FALSE(val.user_value.has_i32());
}
TEST(ProtocolBufferWithRequiredFields, OptionalFieldInSubprotoIsEventuallySet) {
auto domain = Arbitrary<TestProtobufWithRequired>()
.WithRepeatedFieldsMaxSize(0)
.WithProtobufFieldUnset("sub_req");
absl::BitGen bitgen;
Value val(domain, bitgen);
ASSERT_TRUE(val.user_value.IsInitialized()) << val.user_value;
for (int i = 0; i < 1000; ++i) {
val.Mutate(domain, bitgen, {}, false);
ASSERT_TRUE(val.user_value.IsInitialized()) << val.user_value;
if (val.user_value.has_req_sub() &&
val.user_value.req_sub().has_subproto_i32())
break;
}
EXPECT_TRUE(val.user_value.has_req_sub() &&
val.user_value.req_sub().has_subproto_i32());
}
TEST(ProtocolBufferWithRequiredFields,
OptionalFieldInSubprotoIsEventuallyUnset) {
auto domain = Arbitrary<TestProtobufWithRequired>()
.WithRepeatedFieldsMaxSize(0)
.WithProtobufFieldUnset("sub_req");
absl::BitGen bitgen;
Value val(domain, bitgen);
ASSERT_TRUE(val.user_value.IsInitialized()) << val.user_value;
// With the restricted domain, the probability of unsetting the field
// req_sub.subproto_i32 is at least 1/800. Hence, within 11000 iterations
// we'll fail to observe this event with probability at most 10^(-6).
for (int i = 0; i < 11000; ++i) {
val.Mutate(domain, bitgen, {}, false);
ASSERT_TRUE(val.user_value.IsInitialized()) << val.user_value;
if (val.user_value.has_req_sub() &&
!val.user_value.req_sub().has_subproto_i32())
break;
}
EXPECT_TRUE(val.user_value.has_req_sub() &&
!val.user_value.req_sub().has_subproto_i32());
}
bool IsTestProtobufWithRequired(const FieldDescriptor* field) {
return field->message_type()->full_name() ==
"fuzztest.internal.TestProtobufWithRequired";
}
TEST(ProtocolBufferWithRequiredFields,
OptionalFieldWithRequiredFieldsIsEventuallySet) {
auto domain =
Arbitrary<TestProtobufWithRequired>()
.WithRepeatedFieldsMaxSize(0)
.WithProtobufFields(IsTestProtobufWithRequired,
Arbitrary<TestProtobufWithRequired>()
.WithRepeatedFieldsMaxSize(0)
// Disallow recursive nesting beyond depth 1.
.WithProtobufFieldUnset("sub_req"));
absl::BitGen bitgen;
Value val(domain, bitgen);
ASSERT_TRUE(val.user_value.IsInitialized()) << val.user_value;
for (int i = 0; i < 1000; ++i) {
val.Mutate(domain, bitgen, {}, false);
ASSERT_TRUE(val.user_value.IsInitialized()) << val.user_value;
if (val.user_value.has_sub_req()) {
ASSERT_TRUE(val.user_value.sub_req().IsInitialized()) << val.user_value;
break;
}
}
EXPECT_TRUE(val.user_value.has_sub_req());
}
TEST(ProtocolBufferWithRequiredFields, MapFieldIsEventuallyPopulated) {
auto domain =
Arbitrary<TestProtobufWithRequired>()
.WithRepeatedFieldsMaxSize(1)
.WithProtobufFields(IsTestProtobufWithRequired,
Arbitrary<TestProtobufWithRequired>()
.WithRepeatedFieldsMaxSize(0)
// Disallow recursive nesting beyond depth 1.
.WithProtobufFieldUnset("sub_req"));
absl::BitGen bitgen;
Value val(domain, bitgen);
ASSERT_TRUE(val.user_value.IsInitialized()) << val.user_value;
bool found = false;
for (int i = 0; i < 1000 && !found; ++i) {
val.Mutate(domain, bitgen, {}, false);
ASSERT_TRUE(val.user_value.IsInitialized()) << val.user_value;
for (const auto& pair : val.user_value.map_sub_req()) {
found = true;
ASSERT_TRUE(pair.second.IsInitialized()) << pair.second;
}
}
EXPECT_TRUE(found);
}
TEST(ProtocolBufferWithRequiredFields, ShrinkingNeverRemovesRequiredFields) {
auto domain =
Arbitrary<TestProtobufWithRequired>()
.WithRepeatedFieldsMaxSize(1)
.WithProtobufFields(IsTestProtobufWithRequired,
Arbitrary<TestProtobufWithRequired>()
.WithRepeatedFieldsMaxSize(0)
// Disallow recursive nesting beyond depth 1.
.WithProtobufFieldUnset("sub_req"));
absl::BitGen bitgen;
Value val(domain, bitgen);
ASSERT_TRUE(val.user_value.IsInitialized()) << val.user_value;
for (int i = 0; i < 1000; ++i) {
val.Mutate(domain, bitgen, {}, false);
ASSERT_TRUE(val.user_value.IsInitialized()) << val.user_value;
}
const auto is_minimal = [](const auto& v) {
return !v.has_i32() && v.req_i32() == 0 && v.req_e() == 0 &&
!v.req_sub().has_subproto_i32() &&
v.req_sub().subproto_rep_i32().empty() && !v.has_sub_req();
};
while (!is_minimal(val.user_value)) {
val.Mutate(domain, bitgen, {}, true);
ASSERT_TRUE(val.user_value.IsInitialized()) << val.user_value;
}
}
TEST(ProtocolBufferWithRecursiveFields, InfiniteleyRecursiveFieldsAreNotSet) {
auto domain = Arbitrary<internal::TestProtobufWithRepeatedRecursionSubproto>()
.WithRepeatedFieldsAlwaysSet();
absl::BitGen bitgen;
Value val(domain, bitgen);
ASSERT_TRUE(val.user_value.IsInitialized()) << val.user_value;
for (int i = 0; i < 1000; ++i) {
val.Mutate(domain, bitgen, {}, false);
ASSERT_TRUE(val.user_value.IsInitialized()) << val.user_value;
ASSERT_FALSE(val.user_value.has_list()) << val.user_value;
}
}
TEST(ProtocolBuffer, CanUsePerFieldDomains) {
Domain<TestProtobuf> domain =
Arbitrary<TestProtobuf>()
.WithInt32Field("i32", InRange(1, 4))
.WithStringField("str", PrintableAsciiString().WithSize(4))
.WithEnumField(
"e", ElementOf<int>({TestProtobuf::Label2, TestProtobuf::Label4}))
.WithRepeatedBoolField("rep_b", VectorOf(Just(true)).WithSize(2))
.WithProtobufField("subproto",
Arbitrary<TestSubProtobuf>().WithInt32Field(
"subproto_i32", Just(-1)));
absl::BitGen bitgen;
Value val(domain, bitgen);
Set<int32_t> i32_values;
Set<std::string> str_values;
Set<TestProtobuf::Enum> e_values;
Set<std::vector<bool>> rep_b_values;
Set<int> subproto_i32_values;
// InRange(1, 4) -> 4 values.
static constexpr int i32_count = 4;
// There are way too many possible strings, so check we find a handful.
static constexpr int str_count = 10;
// Only two enums in the ElementOf.
static constexpr int e_count = 2;
// Only one possible value: `{true, true}`.
static constexpr int rep_p_count = 1;
// Only one possible value: `-1`
static constexpr int subproto_i32_count = 1;
while (i32_values.size() < i32_count || str_values.size() < str_count ||
e_values.size() < e_count || rep_b_values.size() < rep_p_count ||
subproto_i32_values.size() < subproto_i32_count) {
val.Mutate(domain, bitgen, {}, false);
if (val.user_value.has_i32()) i32_values.insert(val.user_value.i32());
if (val.user_value.has_str()) str_values.insert(val.user_value.str());
if (val.user_value.has_e()) e_values.insert(val.user_value.e());
if (!val.user_value.rep_b().empty()) {
rep_b_values.emplace(val.user_value.rep_b().begin(),
val.user_value.rep_b().end());
}
if (val.user_value.subproto().has_subproto_i32()) {
subproto_i32_values.insert(val.user_value.subproto().subproto_i32());
}
// Let's make sure the custom corpus information can be serialized properly.
auto parsed = domain.ParseCorpus(domain.SerializeCorpus(val.corpus_value));
ASSERT_TRUE(parsed);
auto value_copy = domain.GetValue(*parsed);
EXPECT_TRUE(Eq{}(val.user_value, value_copy));
}
EXPECT_THAT(i32_values, UnorderedElementsAre(1, 2, 3, 4));
EXPECT_THAT(str_values, Each(SizeIs(4)));
EXPECT_THAT(e_values,
UnorderedElementsAre(TestProtobuf::Label2, TestProtobuf::Label4));
EXPECT_THAT(rep_b_values, UnorderedElementsAre(ElementsAre(true, true)));
EXPECT_THAT(subproto_i32_values, ElementsAre(-1));
}
TEST(ProtocolBuffer, InvalidInputReportsError) {
EXPECT_DEATH_IF_SUPPORTED(
Arbitrary<TestProtobuf>().WithStringField("i32",
Arbitrary<std::string>()),
"Failed precondition.*"
"does not match field `fuzztest.internal.TestProtobuf.i32`");
EXPECT_DEATH_IF_SUPPORTED(
Arbitrary<TestProtobuf>()
.WithInt32Field("i32", Just(0))
.WithInt32Field("i32", Just(0)),
"Failed precondition.*"
"field `fuzztest.internal.TestProtobuf.i32` has been set multiple times");
}
TEST(ProtocolBuffer, ValidationRejectsUnexpectedOptionalField) {
TestSubProtobuf user_value;
auto domain_with_optional_always_set =
Arbitrary<TestSubProtobuf>().WithOptionalFieldsAlwaysSet();
auto corpus_value = domain_with_optional_always_set.FromValue(user_value);
EXPECT_THAT(
domain_with_optional_always_set.ValidateCorpusValue(*corpus_value),
IsInvalid("Invalid value for field subproto_i32 >> Optional value must "
"be set"));
auto domain_with_repeated_always_set =
Arbitrary<TestSubProtobuf>().WithRepeatedFieldsAlwaysSet();
EXPECT_THAT(domain_with_repeated_always_set.ValidateCorpusValue(
*domain_with_optional_always_set.FromValue(user_value)),
IsInvalid("Invalid value for field subproto_rep_i32 >> Invalid "
"size: 0. Min size: 1"));
}
TEST(ProtocolBuffer, SerializeAndParseCanHandleExtensions) {
auto domain = Arbitrary<TestProtobufWithExtension>();
TestProtobufWithExtension user_value;
user_value.SetExtension(ProtoExtender::ext, "Hello?!?!");
auto corpus_value = domain.FromValue(user_value);
EXPECT_TRUE(corpus_value != std::nullopt);
auto serialized = domain.SerializeCorpus(corpus_value.value());
auto parsed = domain.ParseCorpus(serialized);
EXPECT_TRUE(parsed != std::nullopt);
auto user_value_after_serialize_parse = domain.GetValue(parsed.value());
EXPECT_EQ("Hello?!?!",
user_value_after_serialize_parse.GetExtension(ProtoExtender::ext));
}
TEST(ProtocolBuffer, ValidationRejectsUnexpectedSingularField) {
absl::BitGen bitgen;
Domain<TestProtobuf> domain_a =
Arbitrary<TestProtobuf>().WithInt32FieldAlwaysSet("i32", InRange(1, 4));
Domain<TestProtobuf> domain_b =
Arbitrary<TestProtobuf>().WithInt32FieldUnset("i32");
Value value_a(domain_a, bitgen);
Value value_b(domain_b, bitgen);
ASSERT_OK(domain_a.ValidateCorpusValue(value_a.corpus_value));
ASSERT_OK(domain_b.ValidateCorpusValue(value_b.corpus_value));
EXPECT_THAT(
domain_a.ValidateCorpusValue(value_b.corpus_value),
IsInvalid("Invalid value for field i32 >> Optional value must be set"));
EXPECT_THAT(
domain_b.ValidateCorpusValue(value_a.corpus_value),
IsInvalid("Invalid value for field i32 >> Optional value must be null"));
}
TEST(ProtocolBuffer, ValidationRejectsUnexpectedSingularExtensionField) {
absl::BitGen bitgen;
Domain<TestProtobufWithExtension> domain_a =
Arbitrary<TestProtobufWithExtension>().WithFieldAlwaysSet(
"fuzztest.internal.ProtoExtender.ext");
Domain<TestProtobufWithExtension> domain_b =
Arbitrary<TestProtobufWithExtension>().WithStringFieldUnset(
"fuzztest.internal.ProtoExtender.ext");
Value value_a(domain_a, bitgen);
Value value_b(domain_b, bitgen);
ASSERT_OK(domain_a.ValidateCorpusValue(value_a.corpus_value));
ASSERT_OK(domain_b.ValidateCorpusValue(value_b.corpus_value));
EXPECT_THAT(domain_a.ValidateCorpusValue(value_b.corpus_value),
IsInvalid(testing::MatchesRegex(
R"(.* field ext .* Optional value must be set)")));
EXPECT_THAT(domain_b.ValidateCorpusValue(value_a.corpus_value),
IsInvalid(testing::MatchesRegex(
R"(.* field ext .* Optional value must be null)")));
}
TEST(ProtocolBuffer, ValidationRejectsUnexpectedRepeatedField) {
absl::BitGen bitgen;
Domain<TestProtobuf> domain_a =
Arbitrary<TestProtobuf>().WithRepeatedInt32Field(
"rep_i32", VectorOf(InRange(1, 4)).WithMinSize(1));
Domain<TestProtobuf> domain_b =
Arbitrary<TestProtobuf>().WithRepeatedInt32Field(
"rep_i32", VectorOf(InRange(1, 4)).WithMaxSize(0));
Value value_a(domain_a, bitgen);
Value value_b(domain_b, bitgen);
ASSERT_OK(domain_a.ValidateCorpusValue(value_a.corpus_value));
ASSERT_OK(domain_b.ValidateCorpusValue(value_b.corpus_value));
EXPECT_THAT(
domain_a.ValidateCorpusValue(value_b.corpus_value),
IsInvalid(testing::MatchesRegex(
R"(Invalid value for field rep_i32 >> Invalid size: .+. Min size: 1)")));
EXPECT_THAT(
domain_b.ValidateCorpusValue(value_a.corpus_value),
IsInvalid(testing::MatchesRegex(
R"(Invalid value for field rep_i32 >> Invalid size: .+. Max size: 0)")));
}
TEST(ProtocolBuffer, ValidationRejectsUnexpectedRepeatedExtensionField) {
absl::BitGen bitgen;
Domain<TestProtobufWithExtension> domain_a =
Arbitrary<TestProtobufWithExtension>().WithRepeatedFieldMinSize(
"fuzztest.internal.ProtoExtender.rep_ext", 1);
Domain<TestProtobufWithExtension> domain_b =
Arbitrary<TestProtobufWithExtension>().WithRepeatedFieldMaxSize(
"fuzztest.internal.ProtoExtender.rep_ext", 0);
Value value_a(domain_a, bitgen);
Value value_b(domain_b, bitgen);
ASSERT_OK(domain_a.ValidateCorpusValue(value_a.corpus_value));
ASSERT_OK(domain_b.ValidateCorpusValue(value_b.corpus_value));
EXPECT_THAT(domain_a.ValidateCorpusValue(value_b.corpus_value),
IsInvalid(testing::MatchesRegex(
R"(.* field rep_ext .* Invalid size: 0. Min size: 1)")));
EXPECT_THAT(domain_b.ValidateCorpusValue(value_a.corpus_value),
IsInvalid(testing::MatchesRegex(
R"(.* field rep_ext .* Invalid size: .+. Max size: 0)")));
}
TEST(ProtocolBuffer, WithFieldsAlwaysSetResetsWithMaxRepeatedFieldsSize) {
absl::BitGen bitgen;
Domain<TestProtobuf> domain =
Arbitrary<TestProtobuf>()
.WithFieldsUnset()
.WithRepeatedFieldsMaxSize(
[](const google::protobuf::FieldDescriptor* field) {
return field->name() == "rep_i32";
},
1)
.WithFieldsAlwaysSet();
EXPECT_THAT(GenerateInitialValues(domain, 1000),
Contains(ResultOf(
[](const Value<Domain<TestProtobuf>>& val) {
return val.user_value.rep_i32_size();
},
Gt(1))));
}
bool IsInt64(const FieldDescriptor* field) {
return field->type() == FieldDescriptor::TYPE_INT64;
}
TEST(ProtocolBuffer, ValidationRejectsIncorrectlySetOneofField) {
Domain<TestProtobuf> domain_a = Arbitrary<TestProtobuf>();
Domain<TestProtobuf> domain_b = Arbitrary<TestProtobuf>()
.WithOneofAlwaysSet("oneof_field")
.WithFieldsUnset(IsInt64)
.WithFieldUnset("oneof_u32");
TestProtobuf user_value_1;
user_value_1.set_oneof_u32(1);
auto corpus_value_1 = domain_a.FromValue(user_value_1);
EXPECT_THAT(
domain_b.ValidateCorpusValue(*corpus_value_1),
IsInvalid(
"Invalid value for field oneof_u32 >> Optional value must be null"));
TestProtobuf user_value_2;
user_value_2.set_oneof_i64(1);
auto corpus_value_2 = domain_a.FromValue(user_value_2);
EXPECT_THAT(
domain_b.ValidateCorpusValue(*corpus_value_2),
IsInvalid(
"Invalid value for field oneof_i64 >> Optional value must be null"));
}
TEST(ProtocolBuffer, ValidationRejectsUnsetOneofsWithOneofAlwaysSet) {
absl::BitGen bitgen;
Domain<TestProtobuf> domain_a = Arbitrary<TestProtobuf>();
Domain<TestProtobuf> domain_b =
Arbitrary<TestProtobuf>().WithOneofAlwaysSet("oneof_field");
TestProtobuf user_value;
auto corpus_value = domain_a.FromValue(user_value);
EXPECT_THAT(domain_b.ValidateCorpusValue(*corpus_value),
IsInvalid("Oneof oneof_field is not set"));
}
TEST(ProtocolBufferEnum, Arbitrary) {
auto domain = Arbitrary<TestProtobuf_Enum>();
absl::BitGen bitgen;
Value val(domain, bitgen);
Set<TestProtobuf_Enum> s;
while (s.size() < internal::TestProtobuf_Enum_descriptor()->value_count()) {
s.insert(val.user_value);
val.Mutate(domain, bitgen, {}, false);
}
val.Mutate(domain, bitgen, {}, true);
}
TEST(ArbitraryProtocolBufferEnum, InitGeneratesSeeds) {
auto domain = Arbitrary<TestProtobuf_Enum>().WithSeeds(
{TestProtobuf_Enum::TestProtobuf_Enum_Label5});
EXPECT_THAT(
GenerateInitialValues(domain, 1000),
Contains(Value(domain, TestProtobuf_Enum::TestProtobuf_Enum_Label5))
// Since there are only 5 enum elements, the seed will surely appear
// at least once. To make the test meaningful, we expect to see it at
// least half the time, unlike the other 4 elements.
.Times(Ge(500)));
}
TEST(ProtocolBuffer, CountNumberOfFieldsCorrect) {
using T = TestProtobuf;
using SubT = TestSubProtobuf;
auto domain = Arbitrary<T>();
T v;
auto corpus_v_uninitialized = domain.FromValue(v);
EXPECT_TRUE(corpus_v_uninitialized != std::nullopt);
EXPECT_EQ(domain.CountNumberOfFields(corpus_v_uninitialized.value()), 26);
v.set_allocated_subproto(new SubT());
auto corpus_v_initizalize_one_optional_proto = domain.FromValue(v);
EXPECT_TRUE(corpus_v_initizalize_one_optional_proto != std::nullopt);
EXPECT_EQ(domain.CountNumberOfFields(
corpus_v_initizalize_one_optional_proto.value()),
28);
v.add_rep_subproto();
auto corpus_v_initizalize_one_repeated_proto_1 = domain.FromValue(v);
EXPECT_TRUE(corpus_v_initizalize_one_repeated_proto_1 != std::nullopt);
EXPECT_EQ(domain.CountNumberOfFields(
corpus_v_initizalize_one_repeated_proto_1.value()),
30);
v.add_rep_subproto();
auto corpus_v_initizalize_one_repeated_proto_2 = domain.FromValue(v);
EXPECT_TRUE(corpus_v_initizalize_one_repeated_proto_2 != std::nullopt);
EXPECT_EQ(domain.CountNumberOfFields(
corpus_v_initizalize_one_repeated_proto_2.value()),
32);
}
auto FieldNameHasSubstr(absl::string_view field_name) {
return [field_name = std::string(field_name)](const FieldDescriptor* field) {
return absl::StrContains(field->name(), field_name);
};
}
TEST(ProtocolBuffer, ProtobufOfIsCustomizable) {
auto domain =
ProtobufOf([] { return &TestProtobuf::default_instance(); })
.WithFieldsAlwaysSet(FieldNameHasSubstr("i32"))
.WithFieldsUnset(FieldNameHasSubstr("i64"))
.WithOptionalFieldsUnset(FieldNameHasSubstr("u32"))
.WithOptionalFieldsAlwaysSet(FieldNameHasSubstr("u64"))
.WithRepeatedFieldsUnset(FieldNameHasSubstr("32"))
.WithRepeatedFieldsAlwaysSet(FieldNameHasSubstr("64"))
.WithRepeatedFieldsMinSize(FieldNameHasSubstr("rep_b"), 3)
.WithRepeatedFieldsMaxSize(FieldNameHasSubstr("rep_b"), 4)
.WithFloatFields(InRange(1.0f, 2.0f))
.WithOptionalDoubleFields(InRange(-1.0, 1.0))
.WithRepeatedDoubleFields(InRange(-1.0, 1.0))
.WithProtobufFields(FieldNameHasSubstr("subproto"),
Arbitrary<TestSubProtobuf>().WithFieldsUnset())
.WithRepeatedProtobufFields(
FieldNameHasSubstr("subproto"),
Arbitrary<TestSubProtobuf>().WithFieldsAlwaysSet());
EXPECT_THAT(
GenerateInitialValues(domain, 1000),
Each(ResultOf(
[](const Value<decltype(domain)>& val) {
auto v =
*dynamic_cast<TestProtobuf*>(val.user_value.get());
for (const auto& s : v.rep_subproto()) {
if (!s.has_subproto_i32()) return false;
if (s.subproto_rep_i32_size() == 0) return false;
}
if (v.has_subproto()) {
auto& s = v.subproto();
if (s.has_subproto_i32()) return false;
if (s.subproto_rep_i32_size() > 0) return false;
}
for (const auto& d : v.rep_d()) {
if (d < -1 || d > 1) return false;
}
if (v.has_d()) {
if (v.d() < -1 || v.d() > 1) return false;
}
if (v.has_f()) {
if (v.f() < 1 || v.f() > 2) return false;
}
if (v.rep_b_size() < 3) return false;
if (v.rep_b_size() > 4) return false;
if (v.rep_i32_size() > 0) return false;
if (v.rep_u32_size() > 0) return false;
if (v.rep_i64_size() == 0) return false;
if (v.rep_u64_size() == 0) return false;
if (v.has_u32()) return false;
if (!v.has_u64()) return false;
if (!v.has_i32()) return false;
if (v.has_i64()) return false;
return true;
},
true)));
}
absl::Duration DoMutations(const absl::SeedSeq& seed_seq) {
auto domain = Arbitrary<internal::TestProtobufWithRecursion>();
absl::BitGen bitgen{seed_seq};
auto corpus_value = domain.Init(bitgen);
absl::Time start = absl::Now();
for (int j = 0; j < 2000; ++j) {
domain.Mutate(corpus_value, bitgen, {}, false);
}
return absl::Now() - start;
}
TEST(ProtocolBuffer, MutationInParallelIsEfficient) {
const unsigned int num_threads =
std::max(1u, std::thread::hardware_concurrency() / 2);
std::cout << "num threads: " << num_threads << "\n";
// Make the mutation runs comparable by using the same seed sequence.
absl::SeedSeq seed_seq = absl::MakeSeedSeq();
const absl::Duration single_thread_time = DoMutations(seed_seq);
std::cout << "total time (single thread): " << single_thread_time << "\n";
std::vector<std::thread> workers;
std::vector<absl::Duration> durations(num_threads);
workers.reserve(num_threads);
for (int i = 0; i < num_threads; ++i) {
workers.emplace_back(
[&durations, &seed_seq, i] { durations[i] = DoMutations(seed_seq); });
}
for (auto& worker : workers) worker.join();
const absl::Duration multi_thread_time =
*std::max_element(durations.begin(), durations.end());
std::cout << "total time (" << num_threads
<< " threads): " << multi_thread_time << "\n";
std::cout << "ratio: " << multi_thread_time / single_thread_time << "\n";
}
} // namespace
} // namespace fuzztest