mediapipe/mediapipe/util/packet_test_util.h
Hadon Nash ea4989b6f1 Internal change
PiperOrigin-RevId: 489135553
2022-11-17 00:07:59 -08:00

182 lines
6.1 KiB
C++

// Copyright 2021 The MediaPipe 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.
//
// Utilities that help to make assertions about packet contents in tests.
#ifndef MEDIAPIPE_UTIL_PACKET_TEST_UTIL_H_
#define MEDIAPIPE_UTIL_PACKET_TEST_UTIL_H_
#include <ostream>
#include <string>
#include <typeinfo>
#include "mediapipe/framework/demangle.h"
#include "mediapipe/framework/packet.h"
#include "mediapipe/framework/port/gmock.h"
#include "mediapipe/framework/port/gtest.h"
#include "mediapipe/framework/timestamp.h"
namespace mediapipe {
namespace internal {
template <typename PayloadType>
class PacketMatcher : public testing::MatcherInterface<const Packet&> {
public:
template <typename InnerMatcher>
explicit PacketMatcher(InnerMatcher inner_matcher)
: inner_matcher_(
testing::SafeMatcherCast<const PayloadType&>(inner_matcher)) {}
// Returns true iff the packet contains value of PayloadType satisfying
// the inner matcher.
bool MatchAndExplain(const Packet& packet,
testing::MatchResultListener* listener) const override {
if (!packet.ValidateAsType<PayloadType>().ok()) {
*listener << packet.DebugString() << " does not contain expected type "
<< ExpectedTypeName();
return false;
}
testing::StringMatchResultListener match_listener;
const PayloadType& payload = packet.Get<PayloadType>();
const bool matches =
inner_matcher_.MatchAndExplain(payload, &match_listener);
const std::string explanation = match_listener.str();
*listener << packet.DebugString() << " containing value "
<< testing::PrintToString(payload);
if (!explanation.empty()) {
*listener << ", which " << explanation;
}
return matches;
}
void DescribeTo(std::ostream* os) const override {
*os << "packet contains value of type " << ExpectedTypeName() << " that ";
inner_matcher_.DescribeTo(os);
}
void DescribeNegationTo(std::ostream* os) const override {
*os << "packet does not contain value of type " << ExpectedTypeName()
<< " that ";
inner_matcher_.DescribeNegationTo(os);
}
private:
static std::string ExpectedTypeName() {
return ::mediapipe::Demangle(typeid(PayloadType).name());
}
const testing::Matcher<const PayloadType&> inner_matcher_;
};
inline std::string SourceString(Timestamp t) {
return (t.IsSpecialValue())
? t.DebugString()
: absl::StrCat("Timestamp(", t.DebugString(), ")");
}
template <typename T>
std::string SourceString(Packet packet) {
std::ostringstream oss;
if (packet.IsEmpty()) {
oss << "Packet()";
} else {
oss << "MakePacket<" << MediaPipeTypeStringOrDemangled<T>() << ">("
<< packet.Get<T>() << ")";
}
oss << ".At(" << SourceString(packet.Timestamp()) << ")";
return oss.str();
}
} // namespace internal
// Creates matcher validating that the packet contains value of expected type
// and satisfying the provided inner matcher.
//
// PayloadType template parameter has to be specified explicitly, but matcher
// type can be inferred. Example:
//
// EXPECT_THAT(MakePacket<int>(42), PacketContains<int>(Eq(42)))
template <typename PayloadType, typename InnerMatcher>
inline testing::Matcher<const Packet&> PacketContains(
InnerMatcher inner_matcher) {
return testing::MakeMatcher(
new internal::PacketMatcher<PayloadType>(inner_matcher));
}
// Creates matcher validating the packet's timestamp satisfies the provided
// timestamp_matcher. It also checks that the packet contains value of expected
// type and satisfies the provided content matcher.
//
// PayloadType template parameter has to be specified explicitly, but matcher
// type can be inferred. Example:
//
// EXPECT_THAT(MakePacket<int>(42).At(Timestamp(20)),
// PacketContainsTimestampAndPayload<int>( //
// Eq(Timestamp(20)),
// Eq(42)))
template <typename PayloadType, typename TimestampMatcher,
typename ContentMatcher>
inline testing::Matcher<const Packet&> PacketContainsTimestampAndPayload(
TimestampMatcher timestamp_matcher, ContentMatcher content_matcher) {
return testing::AllOf(
testing::Property("Packet::Timestamp", &Packet::Timestamp,
timestamp_matcher),
PacketContains<PayloadType>(content_matcher));
}
template <typename T>
class PacketEqMatcher : public testing::MatcherInterface<Packet> {
public:
PacketEqMatcher(Packet packet) : packet_(packet) {}
void DescribeTo(::std::ostream* os) const override {
*os << "The expected packet: " << internal::SourceString<T>(packet_);
}
bool MatchAndExplain(Packet value,
testing::MatchResultListener* listener) const override {
bool unequal = (value.Timestamp() != packet_.Timestamp() ||
value.IsEmpty() != packet_.IsEmpty() ||
(!value.IsEmpty() && value.Get<T>() != packet_.Get<T>()));
if (unequal && listener->IsInterested()) {
*listener << "The actual packet: " << internal::SourceString<T>(value);
}
return !unequal;
}
const Packet packet_;
};
template <typename T>
testing::Matcher<Packet> PacketEq(Packet packet) {
return MakeMatcher(new PacketEqMatcher<T>(packet));
}
template <typename T>
std::vector<testing::Matcher<Packet>> PacketMatchers(
std::vector<Packet> packets) {
std::vector<testing::Matcher<Packet>> result;
for (const auto& packet : packets) {
result.push_back(PacketEq<T>(packet));
}
return result;
}
} // namespace mediapipe
namespace mediapipe {
using mediapipe::PacketContains;
using mediapipe::PacketContainsTimestampAndPayload;
} // namespace mediapipe
#endif // MEDIAPIPE_UTIL_PACKET_TEST_UTIL_H_