123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351 |
- #include <yql/essentials/public/decimal/yql_decimal.h>
- #include <yql/essentials/public/decimal/yql_decimal_serialize.h>
- #include <library/cpp/testing/unittest/registar.h>
- namespace NYql {
- namespace NDecimal {
- Y_UNIT_TEST_SUITE(TYqlDecimalTest) {
- void SimplePositiveTest(TInt128 v, ui8 precision, ui8 scale, const TString& expected) {
- TString result = ToString(v, precision, scale);
- UNIT_ASSERT_VALUES_EQUAL(result, expected);
- TInt128 parsed = FromString(result, precision, scale);
- UNIT_ASSERT(parsed == v);
- }
- void SimpleNegativeFormatTest(TInt128 v, ui8 precision, ui8 scale) {
- TString result = ToString(v, precision, scale);
- UNIT_ASSERT_VALUES_EQUAL(result, "");
- }
- void SimpleSerializeAndDeserialize(TInt128 v, size_t expectedSize) {
- char buff[sizeof(TInt128)];
- const auto s = Serialize(v, buff);
- UNIT_ASSERT_VALUES_EQUAL(s, expectedSize);
- const auto& des = Deserialize(buff, expectedSize);
- UNIT_ASSERT_VALUES_EQUAL(des.second, expectedSize);
- UNIT_ASSERT(des.first == v);
- const auto& e = Deserialize(buff, expectedSize - 1);
- UNIT_ASSERT(e.first == Err());
- }
- template<ui8 Precision, ui8 Scale>
- void CheckMulAndRescale(const TStringBuf& lhs, const TStringBuf& rhs, const TStringBuf& expected) {
- const auto l = FromString(lhs, Precision, Scale);
- const auto r = FromString(rhs, Precision, Scale);
- const auto m = MulAndDivNormalDivider(l, r, GetDivider<Scale>());
- const auto result = ToString(m, Precision, Scale);
- UNIT_ASSERT_VALUES_EQUAL(result, expected);
- }
- template<ui8 Precision, ui8 Scale>
- void CheckDivAndRescale(const TStringBuf& lhs, const TStringBuf& rhs, const TStringBuf& expected) {
- const auto l = FromString(lhs, Precision, Scale);
- const auto r = FromString(rhs, Precision, Scale);
- const auto m = MulAndDivNormalMultiplier(l, GetDivider<Scale>(), r);
- const auto result = ToString(m, Precision, Scale);
- UNIT_ASSERT_VALUES_EQUAL(result, expected);
- }
- template<ui8 Precision, ui8 Scale = 0>
- void CheckMul(const TStringBuf& lhs, const TStringBuf& rhs, const TStringBuf& expected) {
- const auto l = FromString(lhs, Precision, Scale);
- const auto r = FromString(rhs, Precision, Scale);
- const auto m = Mul(l, r);
- const auto result = ToString(m, Precision, Scale);
- UNIT_ASSERT_VALUES_EQUAL(result, expected);
- }
- Y_UNIT_TEST(TestZeroFormat) {
- UNIT_ASSERT_VALUES_EQUAL(ToString(0, 1, 0), "0");
- UNIT_ASSERT_VALUES_EQUAL(ToString(0, 15, 6), "0");
- UNIT_ASSERT_VALUES_EQUAL(ToString(0, 15, 0), "0");
- }
- Y_UNIT_TEST(TestZeroScale) {
- SimplePositiveTest(1, 5, 0, "1");
- SimplePositiveTest(10, 5, 0, "10");
- SimplePositiveTest(100, 5, 0, "100");
- SimplePositiveTest(1000, 5, 0, "1000");
- SimplePositiveTest(10000, 5, 0, "10000");
- SimpleNegativeFormatTest(100000, 5, 0);
- SimpleNegativeFormatTest(1000000, 5, 0);
- // negative numbers
- SimplePositiveTest(-1, 5, 0, "-1");
- SimplePositiveTest(-10, 5, 0, "-10");
- SimplePositiveTest(-100, 5, 0, "-100");
- SimplePositiveTest(-1000, 5, 0, "-1000");
- SimplePositiveTest(-10000, 5, 0, "-10000");
- SimpleNegativeFormatTest(-100000, 5, 0);
- }
- Y_UNIT_TEST(TestFormats) {
- // we have no trailing zeros
- SimplePositiveTest(1, 15, 6, "0.000001");
- SimplePositiveTest(10, 15, 6, "0.00001");
- SimplePositiveTest(100, 15, 6, "0.0001");
- SimplePositiveTest(1000, 15, 6, "0.001");
- SimplePositiveTest(10000, 15, 6, "0.01");
- SimplePositiveTest(100000, 15, 6, "0.1");
- SimplePositiveTest(1000000, 15, 6, "1");
- SimplePositiveTest(10000000, 15, 6, "10");
- SimplePositiveTest(100000000, 15, 6, "100");
- SimplePositiveTest(2020000, 15, 6, "2.02");
- SimplePositiveTest(3003000, 15, 6, "3.003");
- // negative numbers
- SimplePositiveTest(-1, 15, 6, "-0.000001");
- SimplePositiveTest(-10, 15, 6, "-0.00001");
- SimplePositiveTest(-100, 15, 6, "-0.0001");
- SimplePositiveTest(-1000, 15, 6, "-0.001");
- SimplePositiveTest(-10000, 15, 6, "-0.01");
- SimplePositiveTest(-100000, 15, 6, "-0.1");
- SimplePositiveTest(-1000000, 15, 6, "-1");
- SimplePositiveTest(-10000000, 15, 6, "-10");
- SimplePositiveTest(-100000000, 15, 6, "-100");
- SimplePositiveTest(-2020000, 15, 6, "-2.02");
- SimplePositiveTest(-3003000, 15, 6, "-3.003");
- SimplePositiveTest(1, 15, 6, "0.000001");
- SimplePositiveTest(12, 15, 6, "0.000012");
- SimplePositiveTest(123, 15, 6, "0.000123");
- SimplePositiveTest(1234, 15, 6, "0.001234");
- SimplePositiveTest(12345, 15, 6, "0.012345");
- SimplePositiveTest(123456, 15, 6, "0.123456");
- SimplePositiveTest(1234567, 15, 6, "1.234567");
- SimplePositiveTest(12345678, 15, 6, "12.345678");
- SimplePositiveTest(123456789, 15, 6, "123.456789");
- SimplePositiveTest(1234567898, 15, 6, "1234.567898");
- SimplePositiveTest(12345678987ll, 15, 6, "12345.678987");
- SimplePositiveTest(123456789876ll, 15, 6, "123456.789876");
- }
- Y_UNIT_TEST(TestHugeNumberFormat) {
- TInt128 max120 = Inf() - 1;
- const char max120String[] = "99999999999999999999999999999999999"; // 35 digits
- static_assert(sizeof(max120String) == 36, "sizeof(max120String) == 36");
- SimplePositiveTest(max120, MaxPrecision, 0, max120String);
- SimplePositiveTest(max120 + 1, MaxPrecision, 0, "inf");
- TInt128 min120 = -Inf() + 1;
- const char min120String[] = "-99999999999999999999999999999999999";
- static_assert(sizeof(min120String) == 37, "sizeof(min120String) == 37");
- SimplePositiveTest(min120, MaxPrecision, 0, min120String);
- SimplePositiveTest(min120 - 1, MaxPrecision, 0, "-inf");
- // take spot for sign and zero before dot
- const char min120StringAfterDot[] = "-0.99999999999999999999999999999999999"; // 35 by nine + leading zero
- static_assert(sizeof(min120StringAfterDot) == 39, "sizeof(min120StringAfterDot) == 39");
- SimplePositiveTest(min120, MaxPrecision, MaxPrecision, min120StringAfterDot);
- SimpleNegativeFormatTest(1, MaxPrecision + 1, MaxPrecision + 1);
- SimpleNegativeFormatTest(1, MaxPrecision + 1, 0);
- SimpleNegativeFormatTest(1, 2, 3);
- }
- Y_UNIT_TEST(TestFormStringRoundToEven) {
- UNIT_ASSERT(FromString(".51", 1, 0) == 1);
- UNIT_ASSERT(FromString("-0.51", 1, 0) == -1);
- UNIT_ASSERT(FromString("+00000008.5", 1, 0) == 8);
- UNIT_ASSERT(FromString("-8.5000000000000000000000000000000", 1, 0) == -8);
- UNIT_ASSERT(FromString("00008.51", 1, 0) == 9);
- UNIT_ASSERT(FromString("-8.5000000000000000000000000000001", 1, 0) == -9);
- UNIT_ASSERT(FromString("09.499999999999999999999999999999999999999999999999999999999", 1, 0) == 9);
- UNIT_ASSERT(FromString("-9.499999999999999999999999999999999999999999999999999999999", 1, 0) == -9);
- UNIT_ASSERT(FromString("9.50", 2, 0) == 10);
- UNIT_ASSERT(FromString("-9.5", 2, 0) == -10);
- UNIT_ASSERT(FromString("+0.9949", 2, 2) == 99);
- UNIT_ASSERT(FromString("-0.9949", 2, 2) == -99);
- }
- Y_UNIT_TEST(TestInfinityValues) {
- UNIT_ASSERT(FromString("+1", 1, 1) == Inf());
- UNIT_ASSERT(FromString("-1", 1, 1) == -Inf());
- UNIT_ASSERT(FromString("10.000", 1, 0) == Inf());
- UNIT_ASSERT(FromString("-10.000", 1, 0) == -Inf());
- UNIT_ASSERT(FromString("9.500", 1, 0) == Inf());
- UNIT_ASSERT(FromString("-9.500", 1, 0) == -Inf());
- UNIT_ASSERT(FromString("+0.950", 1, 1) == Inf());
- UNIT_ASSERT(FromString("-0.950", 1, 1) == -Inf());
- UNIT_ASSERT(FromString("+0.9950", 2, 2) == Inf());
- UNIT_ASSERT(FromString("-0.9950", 2, 2) == -Inf());
- UNIT_ASSERT(FromString("9999999999999999999999999999999999999.5", 35, 0) == Inf());
- UNIT_ASSERT(FromString("-9999999999999999999999999999999999999.5", 35, 0) == -Inf());
- }
- Y_UNIT_TEST(TestInvalidValues) {
- UNIT_ASSERT(IsValid("+999999999999999991234567890.039493804903849038490312345678909999999999999999990"));
- UNIT_ASSERT(!IsValid("")); // empty
- UNIT_ASSERT(!IsValid("12.2.3")); // double dot
- UNIT_ASSERT(!IsValid("+-12")); // extra sign
- UNIT_ASSERT(!IsValid("463786378O74674")); // letter inside
- UNIT_ASSERT(IsError(FromString("", 35, 15))); // empty
- UNIT_ASSERT(IsError(FromString("12.2.3", 35, 15))); // double dot
- UNIT_ASSERT(IsError(FromString("+-12", 35, 15))); // extra sign
- UNIT_ASSERT(IsError(FromString("463786378O74674", 35, 15))); // letter inside
- UNIT_ASSERT(IsError(FromString("+7.039493804E1", 35, 5))); // letter in tail after scale
- }
- Y_UNIT_TEST(TestFormStringEx) {
- UNIT_ASSERT(FromStringEx("NAN", 13, 1) == Nan());
- UNIT_ASSERT(FromStringEx("+inf", 11, 7) == Inf());
- UNIT_ASSERT(FromStringEx("-inf", 7, 7) == -Inf());
- UNIT_ASSERT(FromStringEx("0.1E3", 10, 1) == 1000);
- UNIT_ASSERT(FromStringEx("0.51e-3", 10, 3) == 1);
- UNIT_ASSERT(FromStringEx("1E30", 10, 0) == Inf());
- UNIT_ASSERT(FromStringEx("1e-30", 10, 0) == 0);
- UNIT_ASSERT(FromStringEx("-1E+99", 10, 2) == -Inf());
- UNIT_ASSERT(FromStringEx("-1e-99", 10, 2) == 0);
- UNIT_ASSERT(FromStringEx("-510e-3", 1, 0) == -1);
- UNIT_ASSERT(FromStringEx("+99E3", 5, 0) == 99000);
- UNIT_ASSERT(FromStringEx("2.1E-130", 35, 2) == 0);
- UNIT_ASSERT(FromStringEx("2.1E0", 35, 2) == 210);
- }
- Y_UNIT_TEST(TestFormStringExInvalidValues) {
- UNIT_ASSERT(IsError(FromStringEx("", 35, 15))); // empty
- UNIT_ASSERT(IsError(FromStringEx("12.2.3", 35, 15))); // double dot
- UNIT_ASSERT(IsError(FromStringEx("+-12", 35, 15))); // extra sign
- UNIT_ASSERT(IsError(FromStringEx("463786378O74674", 35, 15))); // letter inside
- UNIT_ASSERT(IsError(FromStringEx("E2", 35, 15))); // empty
- UNIT_ASSERT(IsError(FromStringEx("E2E4", 35, 15))); // empty
- UNIT_ASSERT(IsError(FromStringEx("NANE5", 35, 15))); // nan with exp
- UNIT_ASSERT(IsError(FromStringEx("infE5", 35, 15))); // inf with exp
- UNIT_ASSERT(IsError(FromStringEx("-infe-5", 35, 15))); // inf with exp
- UNIT_ASSERT(IsError(FromStringEx("2.1E0X", 35, 2))); // not fully parsed exp
- UNIT_ASSERT(IsError(FromStringEx("2.1E+-1", 35, 2))); // two signs
- UNIT_ASSERT(IsError(FromStringEx("ae30", 10, 0))); // bad mantissa
- }
- Y_UNIT_TEST(TestSpecialAsString) {
- UNIT_ASSERT(IsValid("Nan"));
- UNIT_ASSERT(IsValid("INF"));
- UNIT_ASSERT(IsValid("-inf"));
- UNIT_ASSERT_VALUES_EQUAL(ToString(Nan(), 10, 2), "nan");
- UNIT_ASSERT_VALUES_EQUAL(ToString(+Inf(), 10, 2), "inf");
- UNIT_ASSERT_VALUES_EQUAL(ToString(-Inf(), 10, 2), "-inf");
- UNIT_ASSERT(IsNan(FromString("nan", 10, 2)));
- UNIT_ASSERT(IsInf(FromString("+INf", MaxPrecision, 6)));
- UNIT_ASSERT(IsInf(FromString("-inF", 4, 2)));
- }
- Y_UNIT_TEST(TestToStringOfNonNormal) {
- // above Inf
- for (TInt128 i = Inf() + 2, end = Inf() + 100; i < end; i++) {
- UNIT_ASSERT(!IsNormal(i));
- UNIT_ASSERT(ToString(i, MaxPrecision, 0) == nullptr);
- }
- // below -Inf
- for (TInt128 i = -Inf() - 2, end = -Inf() - 100; i < end; i--) {
- UNIT_ASSERT(!IsNormal(i));
- UNIT_ASSERT(ToString(i, MaxPrecision, 0) == nullptr);
- }
- }
- Y_UNIT_TEST(TestSerializeAndDeserialize) {
- SimpleSerializeAndDeserialize(-Nan(), 1U);
- SimpleSerializeAndDeserialize(-Inf(), 1U);
- SimpleSerializeAndDeserialize(-Inf() + 1, 16U);
- SimpleSerializeAndDeserialize(-Inf() + 2, 16U);
- SimpleSerializeAndDeserialize(-65537, 4U);
- SimpleSerializeAndDeserialize(-65536, 3U);
- SimpleSerializeAndDeserialize(-257, 3U);
- SimpleSerializeAndDeserialize(-256, 2U);
- SimpleSerializeAndDeserialize(-3, 2U);
- SimpleSerializeAndDeserialize(-2, 2U);
- SimpleSerializeAndDeserialize(-1, 1U);
- SimpleSerializeAndDeserialize(0, 1U);
- SimpleSerializeAndDeserialize(+1, 2U);
- SimpleSerializeAndDeserialize(+2, 2U);
- SimpleSerializeAndDeserialize(+255, 2U);
- SimpleSerializeAndDeserialize(+256, 3U);
- SimpleSerializeAndDeserialize(+65535, 3U);
- SimpleSerializeAndDeserialize(+65536, 4U);
- SimpleSerializeAndDeserialize(+Inf() - 2, 16U);
- SimpleSerializeAndDeserialize(+Inf() - 1, 16U);
- SimpleSerializeAndDeserialize(+Inf(), 1U);
- SimpleSerializeAndDeserialize(+Nan(), 1U);
- }
- Y_UNIT_TEST(TestMulAndRescale) {
- CheckMulAndRescale<35,35>("0.99999999999999999999999999999999999", "-0.99999999999999999999999999999999999", "-0.99999999999999999999999999999999998");
- CheckMulAndRescale<35,35>("-0.99999999999999999999999999999999999", "0.33333333333333333333333333333333333", "-0.33333333333333333333333333333333333");
- CheckMulAndRescale<35,35>("0.33333333333333333333333333333333333", "0.33333333333333333333333333333333333", "0.11111111111111111111111111111111111");
- CheckMulAndRescale<35,35>("0.99999999999999999999999999999999999", "0.000000000000001", "0.000000000000001");
- CheckMulAndRescale<35,35>("0.99999999999999999999999999999999999", "0.00000000000000101010101", "0.00000000000000101010101");
- CheckMulAndRescale<35,35>("0.12345678901234567890123456789012345", "0.12345678901234567890123456789012345", "0.01524157875323883675049535156256668");
- CheckMulAndRescale<35,34>("9.9999999999999999999999999999999999", "-1.9999999999999999999999999999999999", "-inf");
- CheckMulAndRescale<35,34>("3.3333333333333333333333333333333333", "3.3333333333333333333333333333333333", "inf");
- CheckMulAndRescale<35,34>("3.3333333333333333333333333333333333", "1.3333333333333333333333333333333333", "4.4444444444444444444444444444444443");
- CheckMulAndRescale<35,34>("-1.3333333333333333333333333333333333", "1.3333333333333333333333333333333333", "-1.7777777777777777777777777777777777");
- CheckMulAndRescale<35,34>("-7", "0", "0");
- CheckMulAndRescale<35,34>("inf", "nan", "nan");
- CheckMulAndRescale<35,34>("inf", "0", "nan");
- CheckMulAndRescale<35,34>("-inf", "-inf", "inf");
- }
- Y_UNIT_TEST(TestDivAndRescale) {
- CheckDivAndRescale<35,35>("-0.99999999999999999999999999999999999", "0.33333333333333333333333333333333333", "-inf");
- CheckDivAndRescale<35,35>("0.33333333333333333333333333333333333", "-0.33333333333333333333333333333333333", "-inf");
- CheckDivAndRescale<35,35>("0.12345678901234567890123456789012345", "0.12345678901234567890123456789012345", "inf");
- CheckDivAndRescale<35,34>("9.9999999999999999999999999999999999", "-1.9999999999999999999999999999999999", "-5.0000000000000000000000000000000002");
- CheckDivAndRescale<35,34>("3.3333333333333333333333333333333333", "3.3333333333333333333333333333333333", "1");
- CheckDivAndRescale<35,34>("3.3333333333333333333333333333333333", "1.3333333333333333333333333333333333", "2.5");
- CheckDivAndRescale<35,34>("-1.7777777777777777777777777777777777", "1.3333333333333333333333333333333333", "-1.3333333333333333333333333333333333");
- CheckDivAndRescale<35,34>("-7", "0", "-inf");
- CheckDivAndRescale<35,34>("inf", "0", "inf");
- CheckDivAndRescale<35,34>("inf", "0", "inf");
- CheckDivAndRescale<35,34>("-inf", "inf", "nan");
- }
- Y_UNIT_TEST(TestWideMul) {
- CheckMul<35>("999999999999999", "99999999999999999999", "99999999999999899999000000000000001");
- CheckMul<35>("9999999999999999", "99999999999999999999", "inf");
- CheckMul<35>("-99999999999999999999999999999999999", "10000000000000000000000000000000000", "-inf");
- CheckMul<35>("-99999999999999999999999999999999999", "-1", "99999999999999999999999999999999999");
- CheckMul<35>("-99999999999999999999999999999999999", "-2", "inf");
- CheckMul<35>("nan", "0", "nan");
- CheckMul<35>("inf", "-inf", "-inf");
- CheckMul<35>("inf", "nan", "nan");
- }
- }
- }
- }
|