1
0
Fork 0
mirror of https://github.com/OpenMW/openmw.git synced 2025-01-16 15:29:55 +00:00

Add more tests for printing RefId and clarify some constants

This commit is contained in:
elsid 2023-04-20 00:24:49 +02:00
parent c97df7d770
commit 7db14b3392
No known key found for this signature in database
GPG key ID: 4DE04C198CBA7625
5 changed files with 93 additions and 33 deletions

View file

@ -1,11 +1,17 @@
#include <components/esm/refid.hpp> #include <components/esm/refid.hpp>
#include <gmock/gmock.h>
#include <gtest/gtest.h> #include <gtest/gtest.h>
#include <functional> #include <functional>
#include <map> #include <map>
#include <string> #include <string>
MATCHER(IsPrint, "")
{
return std::isprint(arg) != 0;
}
namespace ESM namespace ESM
{ {
namespace namespace
@ -26,7 +32,7 @@ namespace ESM
TEST(ESMRefIdTest, formIdRefIdIsNotEmpty) TEST(ESMRefIdTest, formIdRefIdIsNotEmpty)
{ {
const RefId refId = RefId::formIdRefId({ 42, 0 }); const RefId refId = RefId::formIdRefId({ .mIndex = 42, .mContentFile = 0 });
EXPECT_FALSE(refId.empty()); EXPECT_FALSE(refId.empty());
} }
@ -53,7 +59,7 @@ namespace ESM
TEST(ESMRefIdTest, defaultConstructedIsNotEqualToFormIdRefId) TEST(ESMRefIdTest, defaultConstructedIsNotEqualToFormIdRefId)
{ {
const RefId a; const RefId a;
const RefId b = RefId::formIdRefId({ 42, 0 }); const RefId b = RefId::formIdRefId({ .mIndex = 42, .mContentFile = 0 });
EXPECT_NE(a, b); EXPECT_NE(a, b);
} }
@ -98,8 +104,8 @@ namespace ESM
TEST(ESMRefIdTest, equalityIsDefinedForFormRefIdAndRefId) TEST(ESMRefIdTest, equalityIsDefinedForFormRefIdAndRefId)
{ {
const FormIdRefId formIdRefId({ 42, 0 }); const FormIdRefId formIdRefId({ .mIndex = 42, .mContentFile = 0 });
const RefId refId = RefId::formIdRefId({ 42, 0 }); const RefId refId = RefId::formIdRefId({ .mIndex = 42, .mContentFile = 0 });
EXPECT_EQ(formIdRefId, refId); EXPECT_EQ(formIdRefId, refId);
} }
@ -125,8 +131,8 @@ namespace ESM
TEST(ESMRefIdTest, lessThanIsDefinedForFormRefIdAndRefId) TEST(ESMRefIdTest, lessThanIsDefinedForFormRefIdAndRefId)
{ {
const FormIdRefId formIdRefId({ 13, 0 }); const FormIdRefId formIdRefId({ .mIndex = 13, .mContentFile = 0 });
const RefId refId = RefId::formIdRefId({ 42, 0 }); const RefId refId = RefId::formIdRefId({ .mIndex = 42, .mContentFile = 0 });
EXPECT_LT(formIdRefId, refId); EXPECT_LT(formIdRefId, refId);
} }
@ -164,14 +170,14 @@ namespace ESM
TEST(ESMRefIdTest, stringRefIdHasStrongOrderWithFormId) TEST(ESMRefIdTest, stringRefIdHasStrongOrderWithFormId)
{ {
const RefId stringRefId = RefId::stringRefId("a"); const RefId stringRefId = RefId::stringRefId("a");
const RefId formIdRefId = RefId::formIdRefId({ 42, 0 }); const RefId formIdRefId = RefId::formIdRefId({ .mIndex = 42, .mContentFile = 0 });
EXPECT_TRUE(stringRefId < formIdRefId); EXPECT_TRUE(stringRefId < formIdRefId);
EXPECT_FALSE(formIdRefId < stringRefId); EXPECT_FALSE(formIdRefId < stringRefId);
} }
TEST(ESMRefIdTest, formIdRefIdHasStrongOrderWithStringView) TEST(ESMRefIdTest, formIdRefIdHasStrongOrderWithStringView)
{ {
const RefId formIdRefId = RefId::formIdRefId({ 42, 0 }); const RefId formIdRefId = RefId::formIdRefId({ .mIndex = 42, .mContentFile = 0 });
const std::string_view stringView = "42"; const std::string_view stringView = "42";
EXPECT_TRUE(stringView < formIdRefId); EXPECT_TRUE(stringView < formIdRefId);
EXPECT_FALSE(formIdRefId < stringView); EXPECT_FALSE(formIdRefId < stringView);
@ -197,7 +203,7 @@ namespace ESM
TEST(ESMRefIdTest, stringRefIdIsNotEqualToFormId) TEST(ESMRefIdTest, stringRefIdIsNotEqualToFormId)
{ {
const RefId stringRefId = RefId::stringRefId("\0"); const RefId stringRefId = RefId::stringRefId("\0");
const RefId formIdRefId = RefId::formIdRefId({ 0, 0 }); const RefId formIdRefId = RefId::formIdRefId({ .mIndex = 0, .mContentFile = 0 });
EXPECT_NE(stringRefId, formIdRefId); EXPECT_NE(stringRefId, formIdRefId);
} }
@ -240,10 +246,19 @@ namespace ESM
{ RefId(), std::string() }, { RefId(), std::string() },
{ RefId::stringRefId("foo"), "foo" }, { RefId::stringRefId("foo"), "foo" },
{ RefId::stringRefId(std::string({ 'a', 0, -1, '\n', '\t' })), { 'a', 0, -1, '\n', '\t' } }, { RefId::stringRefId(std::string({ 'a', 0, -1, '\n', '\t' })), { 'a', 0, -1, '\n', '\t' } },
{ RefId::formIdRefId({ 42, 0 }), "0x2a" }, { RefId::formIdRefId({ .mIndex = 42, .mContentFile = 0 }), "0x2a" },
{ RefId::formIdRefId({ .mIndex = 0xffffff, .mContentFile = std::numeric_limits<std::int32_t>::min() }),
"0xff80000000ffffff" },
{ RefId::formIdRefId({ .mIndex = 0xffffff, .mContentFile = std::numeric_limits<std::int32_t>::max() }),
"0x7fffffffffffff" },
{ RefId::generated(42), "0x2a" }, { RefId::generated(42), "0x2a" },
{ RefId::generated(std::numeric_limits<std::uint64_t>::max()), "0xffffffffffffffff" },
{ RefId::index(REC_ARMO, 42), "ARMO:0x2a" }, { RefId::index(REC_ARMO, 42), "ARMO:0x2a" },
{ RefId::esm3ExteriorCell(-13, 42), "-13:42" }, { RefId::esm3ExteriorCell(-13, 42), "-13:42" },
{ RefId::esm3ExteriorCell(std::numeric_limits<int>::min(), std::numeric_limits<int>::min()),
"-2147483648:-2147483648" },
{ RefId::esm3ExteriorCell(std::numeric_limits<int>::max(), std::numeric_limits<int>::max()),
"2147483647:2147483647" },
}; };
INSTANTIATE_TEST_SUITE_P(ESMRefIdToString, ESMRefIdToStringTest, ValuesIn(toStringParams)); INSTANTIATE_TEST_SUITE_P(ESMRefIdToString, ESMRefIdToStringTest, ValuesIn(toStringParams));
@ -273,10 +288,20 @@ namespace ESM
{ RefId::stringRefId("foo"), "\"foo\"" }, { RefId::stringRefId("foo"), "\"foo\"" },
{ RefId::stringRefId("BAR"), "\"BAR\"" }, { RefId::stringRefId("BAR"), "\"BAR\"" },
{ RefId::stringRefId(std::string({ 'a', 0, -1, '\n', '\t' })), "\"a\\x0\\xFF\\xA\\x9\"" }, { RefId::stringRefId(std::string({ 'a', 0, -1, '\n', '\t' })), "\"a\\x0\\xFF\\xA\\x9\"" },
{ RefId::formIdRefId({ 42, 0 }), "FormId:0x2a" }, { RefId::formIdRefId({ .mIndex = 42, .mContentFile = 0 }), "FormId:0x2a" },
{ RefId::formIdRefId({ .mIndex = 0xffffff, .mContentFile = std::numeric_limits<std::int32_t>::min() }),
"FormId:0xff80000000ffffff" },
{ RefId::formIdRefId({ .mIndex = 0xffffff, .mContentFile = std::numeric_limits<std::int32_t>::max() }),
"FormId:0x7fffffffffffff" },
{ RefId::generated(42), "Generated:0x2a" }, { RefId::generated(42), "Generated:0x2a" },
{ RefId::generated(std::numeric_limits<std::uint64_t>::max()), "Generated:0xffffffffffffffff" },
{ RefId::index(REC_ARMO, 42), "Index:ARMO:0x2a" }, { RefId::index(REC_ARMO, 42), "Index:ARMO:0x2a" },
{ RefId::index(REC_ARMO, std::numeric_limits<std::uint32_t>::max()), "Index:ARMO:0xffffffff" },
{ RefId::esm3ExteriorCell(-13, 42), "Esm3ExteriorCell:-13:42" }, { RefId::esm3ExteriorCell(-13, 42), "Esm3ExteriorCell:-13:42" },
{ RefId::esm3ExteriorCell(std::numeric_limits<int>::min(), std::numeric_limits<int>::min()),
"Esm3ExteriorCell:-2147483648:-2147483648" },
{ RefId::esm3ExteriorCell(std::numeric_limits<int>::max(), std::numeric_limits<int>::max()),
"Esm3ExteriorCell:2147483647:2147483647" },
}; };
INSTANTIATE_TEST_SUITE_P(ESMRefIdToDebugString, ESMRefIdToDebugStringTest, ValuesIn(toDebugStringParams)); INSTANTIATE_TEST_SUITE_P(ESMRefIdToDebugString, ESMRefIdToDebugStringTest, ValuesIn(toDebugStringParams));
@ -300,11 +325,16 @@ namespace ESM
{ RefId::stringRefId("foo"), "foo" }, { RefId::stringRefId("foo"), "foo" },
{ RefId::stringRefId("BAR"), "bar" }, { RefId::stringRefId("BAR"), "bar" },
{ RefId::stringRefId(std::string({ 'a', 0, -1, '\n', '\t' })), { 'a', 0, -1, '\n', '\t' } }, { RefId::stringRefId(std::string({ 'a', 0, -1, '\n', '\t' })), { 'a', 0, -1, '\n', '\t' } },
{ RefId::formIdRefId({ 0, 0 }), "FormId:0x0" }, { RefId::formIdRefId({ .mIndex = 0, .mContentFile = 0 }), "FormId:0x0" },
{ RefId::formIdRefId({ 1, 0 }), "FormId:0x1" }, { RefId::formIdRefId({ .mIndex = 1, .mContentFile = 0 }), "FormId:0x1" },
{ RefId::formIdRefId({ 0x1f, 0 }), "FormId:0x1f" }, { RefId::formIdRefId({ .mIndex = 0x1f, .mContentFile = 0 }), "FormId:0x1f" },
{ RefId::formIdRefId({ 0x1f, 2 }), "FormId:0x200001f" }, { RefId::formIdRefId({ .mIndex = 0x1f, .mContentFile = 2 }), "FormId:0x200001f" },
{ RefId::formIdRefId({ 0xffffff, 0x1abc }), "FormId:0x1abcffffff" }, { RefId::formIdRefId({ .mIndex = 0xffffff, .mContentFile = 0x1abc }), "FormId:0x1abcffffff" },
{ RefId::formIdRefId({ .mIndex = 0xffffff, .mContentFile = std::numeric_limits<std::int32_t>::max() }),
"FormId:0x7fffffffffffff" },
{ RefId::formIdRefId({ .mIndex = 0xffffff, .mContentFile = -1 }), "FormId:0xffffffffffffffff" },
{ RefId::formIdRefId({ .mIndex = 0xffffff, .mContentFile = std::numeric_limits<std::int32_t>::min() }),
"FormId:0xff80000000ffffff" },
{ RefId::generated(0), "Generated:0x0" }, { RefId::generated(0), "Generated:0x0" },
{ RefId::generated(1), "Generated:0x1" }, { RefId::generated(1), "Generated:0x1" },
{ RefId::generated(0x1f), "Generated:0x1f" }, { RefId::generated(0x1f), "Generated:0x1f" },
@ -350,7 +380,7 @@ namespace ESM
template <> template <>
struct GenerateRefId<FormIdRefId> struct GenerateRefId<FormIdRefId>
{ {
static RefId call() { return RefId::formIdRefId({ 42, 0 }); } static RefId call() { return RefId::formIdRefId({ .mIndex = 42, .mContentFile = 0 }); }
}; };
template <> template <>
@ -388,11 +418,27 @@ namespace ESM
{ {
const RefId refId = GenerateRefId<TypeParam>::call(); const RefId refId = GenerateRefId<TypeParam>::call();
const std::string text = refId.serializeText(); const std::string text = refId.serializeText();
for (std::size_t i = 0; i < text.size(); ++i)
ASSERT_TRUE(std::isprint(text[i])) << "index: " << i << ", int value: " << static_cast<int>(text[i]);
EXPECT_EQ(RefId::deserializeText(text), refId); EXPECT_EQ(RefId::deserializeText(text), refId);
} }
TYPED_TEST_P(ESMRefIdTypesTest, serializeTextShouldReturnOnlyPrintableCharacters)
{
const RefId refId = GenerateRefId<TypeParam>::call();
EXPECT_THAT(refId.serializeText(), Each(IsPrint()));
}
TYPED_TEST_P(ESMRefIdTypesTest, toStringShouldReturnOnlyPrintableCharacters)
{
const RefId refId = GenerateRefId<TypeParam>::call();
EXPECT_THAT(refId.toString(), Each(IsPrint()));
}
TYPED_TEST_P(ESMRefIdTypesTest, toDebugStringShouldReturnOnlyPrintableCharacters)
{
const RefId refId = GenerateRefId<TypeParam>::call();
EXPECT_THAT(refId.toDebugString(), Each(IsPrint()));
}
TYPED_TEST_P(ESMRefIdTypesTest, shouldBeEqualToItself) TYPED_TEST_P(ESMRefIdTypesTest, shouldBeEqualToItself)
{ {
const RefId a = GenerateRefId<TypeParam>::call(); const RefId a = GenerateRefId<TypeParam>::call();
@ -416,7 +462,8 @@ namespace ESM
REGISTER_TYPED_TEST_SUITE_P(ESMRefIdTypesTest, serializeThenDeserializeShouldProduceSameValue, REGISTER_TYPED_TEST_SUITE_P(ESMRefIdTypesTest, serializeThenDeserializeShouldProduceSameValue,
serializeTextThenDeserializeTextShouldProduceSameValue, shouldBeEqualToItself, shouldNotBeNotEqualToItself, serializeTextThenDeserializeTextShouldProduceSameValue, shouldBeEqualToItself, shouldNotBeNotEqualToItself,
shouldBeNotLessThanItself); shouldBeNotLessThanItself, serializeTextShouldReturnOnlyPrintableCharacters,
toStringShouldReturnOnlyPrintableCharacters, toDebugStringShouldReturnOnlyPrintableCharacters);
template <class> template <class>
struct RefIdTypes; struct RefIdTypes;

View file

@ -9,23 +9,25 @@ namespace ESM
{ {
std::string ESM3ExteriorCellRefId::toString() const std::string ESM3ExteriorCellRefId::toString() const
{ {
constexpr std::size_t separator = 1;
std::string result; std::string result;
result.resize(getDecIntegralCapacity(mX) + getDecIntegralCapacity(mY) + 3, '\0'); result.resize(getDecIntegralCapacity(mX) + separator + getDecIntegralCapacity(mY), '\0');
const std::size_t endX = serializeDecIntegral(mX, 0, result); const std::size_t endX = serializeDecIntegral(mX, 0, result);
result[endX] = ':'; result[endX] = ':';
const std::size_t endY = serializeDecIntegral(mY, endX + 1, result); const std::size_t endY = serializeDecIntegral(mY, endX + separator, result);
result.resize(endY); result.resize(endY);
return result; return result;
} }
std::string ESM3ExteriorCellRefId::toDebugString() const std::string ESM3ExteriorCellRefId::toDebugString() const
{ {
constexpr std::size_t separator = 1;
std::string result; std::string result;
serializeRefIdPrefix( serializeRefIdPrefix(
getDecIntegralCapacity(mX) + getDecIntegralCapacity(mY) + 1, esm3ExteriorCellRefIdPrefix, result); getDecIntegralCapacity(mX) + separator + getDecIntegralCapacity(mY), esm3ExteriorCellRefIdPrefix, result);
const std::size_t endX = serializeDecIntegral(mX, esm3ExteriorCellRefIdPrefix.size(), result); const std::size_t endX = serializeDecIntegral(mX, esm3ExteriorCellRefIdPrefix.size(), result);
result[endX] = ':'; result[endX] = ':';
const std::size_t endY = serializeDecIntegral(mY, endX + 1, result); const std::size_t endY = serializeDecIntegral(mY, endX + separator, result);
result.resize(endY); result.resize(endY);
return result; return result;
} }

View file

@ -9,7 +9,7 @@ namespace ESM
std::string GeneratedRefId::toString() const std::string GeneratedRefId::toString() const
{ {
std::string result; std::string result;
result.resize(getHexIntegralSize(mValue) + 2, '\0'); result.resize(getHexIntegralSizeWith0x(mValue), '\0');
serializeHexIntegral(mValue, 0, result); serializeHexIntegral(mValue, 0, result);
return result; return result;
} }

View file

@ -9,20 +9,23 @@ namespace ESM
std::string IndexRefId::toString() const std::string IndexRefId::toString() const
{ {
std::string result; std::string result;
result.resize(sizeof(mRecordType) + getHexIntegralSize(mValue) + 3, '\0'); constexpr std::size_t separator = 1;
result.resize(sizeof(mRecordType) + separator + getHexIntegralSizeWith0x(mValue), '\0');
std::memcpy(result.data(), &mRecordType, sizeof(mRecordType)); std::memcpy(result.data(), &mRecordType, sizeof(mRecordType));
result[sizeof(mRecordType)] = ':'; result[sizeof(mRecordType)] = ':';
serializeHexIntegral(mValue, sizeof(mRecordType) + 1, result); serializeHexIntegral(mValue, sizeof(mRecordType) + separator, result);
return result; return result;
} }
std::string IndexRefId::toDebugString() const std::string IndexRefId::toDebugString() const
{ {
std::string result; std::string result;
serializeRefIdPrefix(sizeof(mRecordType) + getHexIntegralSize(mValue) + 1, indexRefIdPrefix, result); constexpr std::size_t separator = 1;
serializeRefIdPrefix(
sizeof(mRecordType) + separator + getHexIntegralSizeWith0x(mValue), indexRefIdPrefix, result);
std::memcpy(result.data() + indexRefIdPrefix.size(), &mRecordType, sizeof(mRecordType)); std::memcpy(result.data() + indexRefIdPrefix.size(), &mRecordType, sizeof(mRecordType));
result[indexRefIdPrefix.size() + sizeof(mRecordType)] = ':'; result[indexRefIdPrefix.size() + sizeof(mRecordType)] = ':';
serializeHexIntegral(mValue, indexRefIdPrefix.size() + sizeof(mRecordType) + 1, result); serializeHexIntegral(mValue, indexRefIdPrefix.size() + sizeof(mRecordType) + separator, result);
return result; return result;
} }

View file

@ -21,9 +21,11 @@ namespace ESM
{ {
if (value == 0) if (value == 0)
return 1; return 1;
constexpr std::size_t lastDigit = 1;
if (value > 0) if (value > 0)
return static_cast<std::size_t>(std::numeric_limits<T>::digits10); return static_cast<std::size_t>(std::numeric_limits<T>::digits10) + lastDigit;
return static_cast<std::size_t>(std::numeric_limits<T>::digits10) + 1; constexpr std::size_t sign = 1;
return static_cast<std::size_t>(std::numeric_limits<T>::digits10) + lastDigit + sign;
} }
template <class T> template <class T>
@ -40,9 +42,15 @@ namespace ESM
return result; return result;
} }
template <class T>
std::size_t getHexIntegralSizeWith0x(T value)
{
return 2 + getHexIntegralSize(value);
}
inline void serializeRefIdPrefix(std::size_t valueSize, std::string_view prefix, std::string& out) inline void serializeRefIdPrefix(std::size_t valueSize, std::string_view prefix, std::string& out)
{ {
out.resize(prefix.size() + valueSize + 2, '\0'); out.resize(prefix.size() + valueSize, '\0');
std::memcpy(out.data(), prefix.data(), prefix.size()); std::memcpy(out.data(), prefix.data(), prefix.size());
} }
@ -70,7 +78,7 @@ namespace ESM
void serializeRefIdValue(T value, std::string_view prefix, std::string& out) void serializeRefIdValue(T value, std::string_view prefix, std::string& out)
{ {
static_assert(!std::is_signed_v<T>); static_assert(!std::is_signed_v<T>);
serializeRefIdPrefix(getHexIntegralSize(value), prefix, out); serializeRefIdPrefix(getHexIntegralSizeWith0x(value), prefix, out);
serializeHexIntegral(value, prefix.size(), out); serializeHexIntegral(value, prefix.size(), out);
} }