mirror of
https://gitlab.com/mbunkus/mkvtoolnix.git
synced 2025-01-07 02:34:53 +00:00
208 lines
6.6 KiB
C++
208 lines
6.6 KiB
C++
#include "common/common_pch.h"
|
|
|
|
#include <iostream>
|
|
|
|
#include <ebml/EbmlBinary.h>
|
|
#include <ebml/EbmlDate.h>
|
|
#include <ebml/EbmlFloat.h>
|
|
#include <ebml/EbmlSInteger.h>
|
|
#include <ebml/EbmlString.h>
|
|
#include <ebml/EbmlUInteger.h>
|
|
#include <ebml/EbmlUnicodeString.h>
|
|
#include <ebml/EbmlVoid.h>
|
|
|
|
#include "common/strings/editing.h"
|
|
#include "common/at_scope_exit.h"
|
|
#include "tests/unit/util.h"
|
|
|
|
namespace mtxut {
|
|
|
|
void
|
|
dump(EbmlElement *element,
|
|
bool with_values,
|
|
unsigned int level) {
|
|
std::string indent_str, value_str;
|
|
size_t i;
|
|
|
|
for (i = 1; i <= level; ++i)
|
|
indent_str += " ";
|
|
|
|
if (with_values) {
|
|
if (dynamic_cast<EbmlUInteger *>(element))
|
|
value_str = to_string(uint64(*static_cast<EbmlUInteger *>(element)));
|
|
|
|
else if (dynamic_cast<EbmlSInteger *>(element))
|
|
value_str = to_string(int64(*static_cast<EbmlSInteger *>(element)));
|
|
|
|
else if (dynamic_cast<EbmlFloat *>(element))
|
|
value_str = to_string(double(*static_cast<EbmlFloat *>(element)));
|
|
|
|
else if (dynamic_cast<EbmlUnicodeString *>(element))
|
|
value_str = UTFstring(*static_cast<EbmlUnicodeString *>(element)).GetUTF8();
|
|
|
|
else if (dynamic_cast<EbmlString *>(element))
|
|
value_str = std::string(*static_cast<EbmlString *>(element));
|
|
|
|
else if (dynamic_cast<EbmlDate *>(element))
|
|
value_str = to_string(static_cast<EbmlDate *>(element)->GetEpochDate());
|
|
|
|
else
|
|
value_str = (boost::format("(type: %1%)") %
|
|
( dynamic_cast<EbmlBinary *>(element) ? "binary"
|
|
: dynamic_cast<EbmlMaster *>(element) ? "master"
|
|
: dynamic_cast<EbmlVoid *>(element) ? "void"
|
|
: "unknown")).str();
|
|
|
|
value_str = " " + value_str;
|
|
}
|
|
|
|
std::cout << indent_str << EBML_NAME(element) << value_str << std::endl;
|
|
|
|
EbmlMaster *master = dynamic_cast<EbmlMaster *>(element);
|
|
if (!master)
|
|
return;
|
|
|
|
for (auto el : *master)
|
|
dump(el, with_values, level + 1);
|
|
}
|
|
|
|
//
|
|
// ----------------------------------------------------------------------
|
|
//
|
|
|
|
::testing::AssertionResult
|
|
EbmlEquals(char const *a_expr,
|
|
char const *b_expr,
|
|
EbmlElement &a,
|
|
EbmlElement &b) {
|
|
std::string error;
|
|
if (ebml_equals_c::check(a, b, error))
|
|
return ::testing::AssertionSuccess();
|
|
else
|
|
return ::testing::AssertionFailure() << a_expr << " and " << b_expr << " are not equal: " << error;
|
|
}
|
|
|
|
//
|
|
// ----------------------------------------------------------------------
|
|
//
|
|
|
|
ebml_equals_c::ebml_equals_c() {
|
|
}
|
|
|
|
|
|
std::string
|
|
ebml_equals_c::get_error() const {
|
|
return m_error;
|
|
}
|
|
|
|
bool
|
|
ebml_equals_c::set_error(std::string const &error,
|
|
EbmlElement *cmp) {
|
|
auto full_path = m_path;
|
|
if (cmp)
|
|
full_path.push_back(EBML_NAME(cmp));
|
|
|
|
m_error = (boost::format("%1% path: %2%") % error % boost::join(full_path, " -> ")).str();
|
|
return false;
|
|
}
|
|
|
|
bool
|
|
ebml_equals_c::set_error(boost::format const &error,
|
|
EbmlElement *cmp) {
|
|
return set_error(error.str(), cmp);
|
|
}
|
|
|
|
bool
|
|
ebml_equals_c::check(EbmlElement &a,
|
|
EbmlElement &b,
|
|
std::string &error) {
|
|
ebml_equals_c checker;
|
|
bool result = checker.compare_impl(a, b);
|
|
error = checker.get_error();
|
|
return result;
|
|
}
|
|
|
|
bool
|
|
ebml_equals_c::compare_impl(EbmlElement &a,
|
|
EbmlElement &b) {
|
|
if (&a == &b)
|
|
return true;
|
|
|
|
if (std::string{EBML_NAME(&a)} != std::string{EBML_NAME(&b)})
|
|
return set_error(boost::format("types are differing: %1% vs %2%") % EBML_NAME(&a) % EBML_NAME(&b));
|
|
|
|
EbmlUInteger *ui_a;
|
|
EbmlSInteger *si_a;
|
|
EbmlFloat *f_a;
|
|
EbmlString *str_a;
|
|
EbmlUnicodeString *ustr_a;
|
|
EbmlMaster *m_a;
|
|
EbmlDate *d_a;
|
|
EbmlBinary *b_a;
|
|
|
|
if ((ui_a = dynamic_cast<EbmlUInteger *>(&a))) {
|
|
auto val_b = uint64(*dynamic_cast<EbmlUInteger *>(&b));
|
|
return uint64(*ui_a) == val_b ? true : set_error(boost::format("UInteger values differ: %1% vs %2%") % uint64(*ui_a) % val_b, &a);
|
|
|
|
} else if ((si_a = dynamic_cast<EbmlSInteger *>(&a))) {
|
|
auto val_b = int64(*dynamic_cast<EbmlSInteger *>(&b));
|
|
return int64(*si_a) == val_b ? true : set_error(boost::format("SInteger values differ: %1% vs %2%") % int64(*si_a) % val_b, &a);
|
|
|
|
} else if ((d_a = dynamic_cast<EbmlDate *>(&a))) {
|
|
auto val_a = d_a->GetEpochDate();
|
|
auto val_b = dynamic_cast<EbmlDate *>(&b)->GetEpochDate();
|
|
return val_a == val_b ? true : set_error(boost::format("Date values differ: %1% vs %2%") % val_a % val_b, &a);
|
|
|
|
} else if ((f_a = dynamic_cast<EbmlFloat *>(&a))) {
|
|
auto val_b = double(*dynamic_cast<EbmlFloat *>(&b));
|
|
return double(*f_a) == val_b ? true : set_error(boost::format("Float values differ: %1% vs %2%") % double(*f_a) % val_b, &a);
|
|
|
|
} else if ((str_a = dynamic_cast<EbmlString *>(&a))) {
|
|
auto val_b = std::string(*dynamic_cast<EbmlString *>(&b));
|
|
return std::string(*str_a) == val_b ? true : set_error(boost::format("String values differ: %1% vs %2%") % std::string(*str_a) % val_b, &a);
|
|
|
|
} else if ((ustr_a = dynamic_cast<EbmlUnicodeString *>(&a))) {
|
|
auto val_a = UTFstring(*dynamic_cast<EbmlUnicodeString *>(&a)).GetUTF8();
|
|
auto val_b = UTFstring(*dynamic_cast<EbmlUnicodeString *>(&b)).GetUTF8();
|
|
return val_a == val_b ? true : set_error(boost::format("UnicodeString values differ: %1% vs %2%") % val_a % val_b, &a);
|
|
|
|
} else if ((b_a = dynamic_cast<EbmlBinary *>(&a))) {
|
|
auto b_b = dynamic_cast<EbmlBinary *>(&b);
|
|
return *b_a == *b_b ? true : set_error(boost::format("Binary values differ; sizes %1% vs %2%") % b_a->GetSize() % b_b->GetSize(), &a);
|
|
|
|
} else if ((m_a = dynamic_cast<EbmlMaster *>(&a))) {
|
|
m_path.push_back(EBML_NAME(&a));
|
|
mtx::at_scope_exit_c popper{[&]() { m_path.pop_back(); }};
|
|
|
|
auto m_b = dynamic_cast<EbmlMaster *>(&b);
|
|
for (int i = 0; i < std::min<int>(m_a->ListSize(), m_b->ListSize()); ++i)
|
|
if (!compare_impl(*(*m_a)[i], *(*m_b)[i]))
|
|
return false;
|
|
|
|
if (m_a->ListSize() == m_b->ListSize())
|
|
return true;
|
|
|
|
std::string op;
|
|
EbmlMaster *with_more, *with_less;
|
|
if (m_a->ListSize() > m_b->ListSize()) {
|
|
op = "more";
|
|
with_more = m_a;
|
|
with_less = m_b;
|
|
} else {
|
|
op = "less";
|
|
with_less = m_a;
|
|
with_more = m_b;
|
|
}
|
|
|
|
std::stringstream error{ (boost::format("LHS contains %1% elements than RHS (%2% vs %3%):") % op % m_a->ListSize() % m_b->ListSize()).str() };
|
|
for (auto i = with_less->ListSize(); i < with_more->ListSize(); ++i)
|
|
error << " " << EBML_NAME((*with_more)[i]);
|
|
|
|
return set_error(error.str());
|
|
|
|
} else
|
|
return set_error(boost::format("unsupported types: %1% and %2%") % EBML_NAME(&a) % EBML_NAME(&b));
|
|
}
|
|
|
|
}
|