summaryrefslogtreecommitdiffstats
path: root/src/test/strtol.cc
diff options
context:
space:
mode:
Diffstat (limited to 'src/test/strtol.cc')
-rw-r--r--src/test/strtol.cc649
1 files changed, 649 insertions, 0 deletions
diff --git a/src/test/strtol.cc b/src/test/strtol.cc
new file mode 100644
index 000000000..ec3f6715b
--- /dev/null
+++ b/src/test/strtol.cc
@@ -0,0 +1,649 @@
+// -*- mode:C++; tab-width:8; c-basic-offset:2; indent-tabs-mode:t -*-
+// vim: ts=8 sw=2 smarttab
+/*
+ * Ceph - scalable distributed file system
+ *
+ * Copyright (C) 2011 Dreamhost
+ *
+ * This is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License version 2.1, as published by the Free Software
+ * Foundation. See file COPYING.
+ *
+ */
+
+#include <cmath>
+#include <string>
+#include <map>
+
+#include "common/strtol.h"
+
+#include "gtest/gtest.h"
+
+static void test_strict_strtoll(const char *str, long long expected, int base)
+{
+ std::string err;
+ long long val = strict_strtoll(str, base, &err);
+ if (!err.empty()) {
+ ASSERT_EQ(err, "");
+ }
+ else {
+ ASSERT_EQ(val, expected);
+ }
+}
+
+static void test_strict_strtol(const char *str, long expected)
+{
+ std::string err;
+ long val = strict_strtol(str, 10, &err);
+ if (!err.empty()) {
+ ASSERT_EQ(err, "");
+ }
+ else {
+ ASSERT_EQ(val, expected);
+ }
+}
+
+static void test_strict_strtod(const char *str, double expected)
+{
+ std::string err;
+ double val = strict_strtod(str, &err);
+ if (!err.empty()) {
+ ASSERT_EQ(err, "");
+ }
+ else {
+ // when comparing floats, use a margin of error
+ if ((expected - 0.001 > val) || (expected + 0.001 < val)) {
+ ASSERT_EQ(val, expected);
+ }
+ }
+}
+
+static void test_strict_strtof(const char *str, float expected)
+{
+ std::string err;
+ float val = strict_strtof(str, &err);
+ if (!err.empty()) {
+ ASSERT_EQ(err, "");
+ }
+ else {
+ // when comparing floats, use a margin of error
+ if ((expected - 0.001 > val) || (expected + 0.001 < val)) {
+ ASSERT_EQ(val, expected);
+ }
+ }
+}
+
+TEST(StrToL, Simple1) {
+ test_strict_strtoll("123", 123, 10);
+ test_strict_strtoll("0", 0, 10);
+ test_strict_strtoll("-123", -123, 10);
+ test_strict_strtoll("8796093022208", 8796093022208LL, 10);
+ test_strict_strtoll("-8796093022208", -8796093022208LL, 10);
+ test_strict_strtoll("123", 123, 0);
+ test_strict_strtoll("0x7b", 123, 0);
+ test_strict_strtoll("4d2", 1234, 16);
+
+ test_strict_strtol("208", 208);
+ test_strict_strtol("-4", -4);
+ test_strict_strtol("0", 0);
+ test_strict_strtol("2147483646", 2147483646);
+
+ test_strict_strtof("0.05", 0.05);
+ test_strict_strtof("0", 0.0);
+ test_strict_strtof("-0", 0.0);
+ test_strict_strtof("10000000.5", 10000000.5);
+
+ test_strict_strtod("-0.2", -0.2);
+ test_strict_strtod("0.1", 0.1);
+ test_strict_strtod("0", 0.0);
+}
+
+static void test_strict_strtoll_err(const char *str)
+{
+ std::string err;
+ strict_strtoll(str, 10, &err);
+ ASSERT_NE(err, "");
+}
+
+static void test_strict_strtol_err(const char *str)
+{
+ std::string err;
+ strict_strtol(str, 10, &err);
+ ASSERT_NE(err, "");
+}
+
+static void test_strict_strtod_err(const char *str)
+{
+ std::string err;
+ strict_strtod(str, &err);
+ ASSERT_NE(err, "");
+}
+
+static void test_strict_strtof_err(const char *str)
+{
+ std::string err;
+ strict_strtof(str, &err);
+ ASSERT_NE(err, "");
+}
+
+TEST(StrToL, Error1) {
+ test_strict_strtoll_err("604462909807314587353088"); // overflow
+ test_strict_strtoll_err("aw shucks"); // invalid
+ test_strict_strtoll_err("343245 aw shucks"); // invalid chars at end
+ test_strict_strtoll_err("-"); // invalid
+
+ test_strict_strtol_err("35 aw shucks"); // invalid chars at end
+ test_strict_strtol_err("--0");
+ test_strict_strtol_err("-");
+
+ test_strict_strtod_err("345345.0-");
+ test_strict_strtod_err("34.0 garbo");
+
+ test_strict_strtof_err("0.05.0");
+}
+
+
+static void test_strict_iecstrtoll(const char *str)
+{
+ std::string err;
+ strict_iecstrtoll(str, &err);
+ ASSERT_EQ(err, "");
+}
+
+static void test_strict_iecstrtoll_units(const std::string& foo,
+ std::string u, const int m)
+{
+ std::string s(foo);
+ s.append(u);
+ const char *str = s.c_str();
+ std::string err;
+ uint64_t r = strict_iecstrtoll(str, &err);
+ ASSERT_EQ(err, "");
+
+ str = foo.c_str();
+ std::string err2;
+ long long tmp = strict_strtoll(str, 10, &err2);
+ ASSERT_EQ(err2, "");
+ tmp = (tmp << m);
+ ASSERT_EQ(tmp, (long long)r);
+}
+
+TEST(IECStrToLL, WithUnits) {
+ std::map<std::string,int> units;
+ units["B"] = 0;
+ units["K"] = 10;
+ units["M"] = 20;
+ units["G"] = 30;
+ units["T"] = 40;
+ units["P"] = 50;
+ units["E"] = 60;
+ units["Ki"] = 10;
+ units["Mi"] = 20;
+ units["Gi"] = 30;
+ units["Ti"] = 40;
+ units["Pi"] = 50;
+ units["Ei"] = 60;
+
+ for (std::map<std::string,int>::iterator p = units.begin();
+ p != units.end(); ++p) {
+ // the upper bound of uint64_t is 2^64 = 4E
+ test_strict_iecstrtoll_units("4", p->first, p->second);
+ test_strict_iecstrtoll_units("1", p->first, p->second);
+ test_strict_iecstrtoll_units("0", p->first, p->second);
+ }
+}
+
+TEST(IECStrToLL, WithoutUnits) {
+ test_strict_iecstrtoll("1024");
+ test_strict_iecstrtoll("1152921504606846976");
+ test_strict_iecstrtoll("0");
+}
+
+static void test_strict_iecstrtoll_err(const char *str)
+{
+ std::string err;
+ strict_iecstrtoll(str, &err);
+ ASSERT_NE(err, "");
+}
+
+TEST(IECStrToLL, Error) {
+ test_strict_iecstrtoll_err("1024F");
+ test_strict_iecstrtoll_err("QDDSA");
+ test_strict_iecstrtoll_err("1b");
+ test_strict_iecstrtoll_err("100k");
+ test_strict_iecstrtoll_err("1000m");
+ test_strict_iecstrtoll_err("1g");
+ test_strict_iecstrtoll_err("20t");
+ test_strict_iecstrtoll_err("100p");
+ test_strict_iecstrtoll_err("1000e");
+ test_strict_iecstrtoll_err("B");
+ test_strict_iecstrtoll_err("M");
+ test_strict_iecstrtoll_err("BM");
+ test_strict_iecstrtoll_err("B0wef");
+ test_strict_iecstrtoll_err("0m");
+ test_strict_iecstrtoll_err("-1"); // it returns uint64_t
+ test_strict_iecstrtoll_err("-1K");
+ test_strict_iecstrtoll_err("1Bi");
+ test_strict_iecstrtoll_err("Bi");
+ test_strict_iecstrtoll_err("bi");
+ test_strict_iecstrtoll_err("gi");
+ test_strict_iecstrtoll_err("100ki");
+ test_strict_iecstrtoll_err("1000mi");
+ test_strict_iecstrtoll_err("1gi");
+ test_strict_iecstrtoll_err("20ti");
+ test_strict_iecstrtoll_err("100pi");
+ test_strict_iecstrtoll_err("1000ei");
+ // the upper bound of uint64_t is 2^64 = 4E, so 1024E overflows
+ test_strict_iecstrtoll_err("1024E"); // overflows after adding the suffix
+}
+
+// since strict_iecstrtoll is an alias of strict_iec_cast<uint64_t>(), quite a few
+// of cases are covered by existing test cases of strict_iecstrtoll already.
+TEST(StrictIECCast, Error) {
+ {
+ std::string err;
+ // the SI prefix is way too large for `int`.
+ (void)strict_iec_cast<int>("2E", &err);
+ ASSERT_NE(err, "");
+ }
+ {
+ std::string err;
+ (void)strict_iec_cast<int>("-2E", &err);
+ ASSERT_NE(err, "");
+ }
+ {
+ std::string err;
+ (void)strict_iec_cast<int>("1T", &err);
+ ASSERT_NE(err, "");
+ }
+ {
+ std::string err;
+ (void)strict_iec_cast<int64_t>("2E", &err);
+ ASSERT_EQ(err, "");
+ }
+ {
+ std::string err;
+ (void)strict_iec_cast<int64_t>("-2E", &err);
+ ASSERT_EQ(err, "");
+ }
+ {
+ std::string err;
+ (void)strict_iec_cast<int64_t>("1T", &err);
+ ASSERT_EQ(err, "");
+ }
+}
+
+
+static void test_strict_sistrtoll(const char *str)
+{
+ std::string err;
+ strict_si_cast<uint64_t>(str, &err);
+ ASSERT_EQ(err, "");
+}
+
+static void test_strict_sistrtoll_units(const std::string& foo,
+ std::string u, const long long m)
+{
+ std::string s(foo);
+ s.append(u);
+ const char *str = s.c_str();
+ std::string err;
+ uint64_t r = strict_si_cast<uint64_t>(str, &err);
+ ASSERT_EQ(err, "");
+
+ str = foo.c_str();
+ std::string err2;
+ long long tmp = strict_strtoll(str, 10, &err2);
+ ASSERT_EQ(err2, "");
+ tmp = (tmp * m);
+ ASSERT_EQ(tmp, (long long)r);
+}
+
+TEST(SIStrToLL, WithUnits) {
+ std::map<std::string,long long> units;
+ units["K"] = pow(10, 3);
+ units["M"] = pow(10, 6);
+ units["G"] = pow(10, 9);
+ units["T"] = pow(10, 12);
+ units["P"] = pow(10, 15);
+ units["E"] = pow(10, 18);
+
+ for (std::map<std::string,long long>::iterator p = units.begin();
+ p != units.end(); ++p) {
+ // the upper bound of uint64_t is 2^64 = 4E
+ test_strict_sistrtoll_units("4", p->first, p->second);
+ test_strict_sistrtoll_units("1", p->first, p->second);
+ test_strict_sistrtoll_units("0", p->first, p->second);
+ }
+}
+
+TEST(SIStrToLL, WithoutUnits) {
+ test_strict_sistrtoll("1024");
+ test_strict_sistrtoll("1152921504606846976");
+ test_strict_sistrtoll("0");
+}
+
+static void test_strict_sistrtoll_err(const char *str)
+{
+ std::string err;
+ strict_si_cast<uint64_t>(str, &err);
+ ASSERT_NE(err, "");
+}
+
+TEST(SIStrToLL, Error) {
+ test_strict_sistrtoll_err("1024F");
+ test_strict_sistrtoll_err("QDDSA");
+ test_strict_sistrtoll_err("1b");
+ test_strict_sistrtoll_err("100k");
+ test_strict_sistrtoll_err("1000m");
+ test_strict_sistrtoll_err("1g");
+ test_strict_sistrtoll_err("20t");
+ test_strict_sistrtoll_err("100p");
+ test_strict_sistrtoll_err("1000e");
+ test_strict_sistrtoll_err("B");
+ test_strict_sistrtoll_err("M");
+ test_strict_sistrtoll_err("BM");
+ test_strict_sistrtoll_err("B0wef");
+ test_strict_sistrtoll_err("0m");
+ test_strict_sistrtoll_err("-1"); // it returns uint64_t
+ test_strict_sistrtoll_err("-1K");
+ test_strict_sistrtoll_err("1Bi");
+ test_strict_sistrtoll_err("Bi");
+ test_strict_sistrtoll_err("bi");
+ test_strict_sistrtoll_err("gi");
+ test_strict_sistrtoll_err("100ki");
+ test_strict_sistrtoll_err("1000mi");
+ test_strict_sistrtoll_err("1gi");
+ test_strict_sistrtoll_err("20ti");
+ test_strict_sistrtoll_err("100pi");
+ test_strict_sistrtoll_err("1000ei");
+ test_strict_sistrtoll_err("1B");
+ // the upper bound of uint64_t is 2^64 = 4E, so 1024E overflows
+ test_strict_sistrtoll_err("1024E"); // overflows after adding the suffix
+}
+
+// since strict_sistrtoll is an alias of strict_si_cast<uint64_t>(), quite a few
+// of cases are covered by existing test cases of strict_sistrtoll already.
+TEST(StrictSICast, Error) {
+ {
+ std::string err;
+ // the SI prefix is way too large for `int`.
+ (void)strict_si_cast<int>("2E", &err);
+ ASSERT_NE(err, "");
+ }
+ {
+ std::string err;
+ (void)strict_si_cast<int>("-2E", &err);
+ ASSERT_NE(err, "");
+ }
+ {
+ std::string err;
+ (void)strict_si_cast<int>("1T", &err);
+ ASSERT_NE(err, "");
+ }
+ {
+ std::string err;
+ (void)strict_si_cast<int64_t>("2E", &err);
+ ASSERT_EQ(err, "");
+ }
+ {
+ std::string err;
+ (void)strict_si_cast<int64_t>("-2E", &err);
+ ASSERT_EQ(err, "");
+ }
+ {
+ std::string err;
+ (void)strict_si_cast<int64_t>("1T", &err);
+ ASSERT_EQ(err, "");
+ }
+}
+
+
+using ceph::parse;
+using ceph::consume;
+using namespace std::literals;
+
+template<typename T>
+inline void test_parse() {
+ auto r = parse<T>("23"sv);
+ ASSERT_TRUE(r);
+ EXPECT_EQ(*r, 23);
+
+ r = parse<T>(" 23"sv);
+ EXPECT_FALSE(r);
+
+ r = parse<T>("-5"sv);
+ if constexpr (std::is_signed_v<T>) {
+ ASSERT_TRUE(r);
+ EXPECT_EQ(*r, -5);
+ } else {
+ EXPECT_FALSE(r);
+ }
+
+ r = parse<T>("meow"sv);
+ EXPECT_FALSE(r);
+
+ r = parse<T>("9yards"sv);
+ EXPECT_FALSE(r);
+}
+
+TEST(Parse, Char) {
+ test_parse<char>();
+}
+
+TEST(Parse, UChar) {
+ test_parse<unsigned char>();
+}
+
+TEST(Parse, SChar) {
+ test_parse<signed char>();
+}
+
+TEST(Parse, UInt8) {
+ test_parse<std::uint8_t>();
+}
+
+TEST(Parse, Int8) {
+ test_parse<std::int8_t>();
+}
+
+TEST(Parse, UInt16) {
+ test_parse<std::uint16_t>();
+}
+
+TEST(Parse, Int16) {
+ test_parse<std::int16_t>();
+}
+
+TEST(Parse, UInt32) {
+ test_parse<std::uint32_t>();
+}
+
+TEST(Parse, Int32) {
+ test_parse<std::int32_t>();
+}
+
+TEST(Parse, UInt64) {
+ test_parse<std::uint64_t>();
+}
+
+TEST(Parse, Int64) {
+ test_parse<std::int64_t>();
+}
+
+TEST(Parse, UIntMax) {
+ test_parse<std::uintmax_t>();
+}
+
+TEST(Parse, IntMax) {
+ test_parse<std::intmax_t>();
+}
+
+TEST(Parse, UIntPtr) {
+ test_parse<std::uintptr_t>();
+}
+
+TEST(Parse, IntPtr) {
+ test_parse<std::intptr_t>();
+}
+
+TEST(Parse, UShort) {
+ test_parse<unsigned short>();
+}
+
+TEST(Parse, Short) {
+ test_parse<short>();
+}
+
+TEST(Parse, ULong) {
+ test_parse<unsigned long>();
+}
+
+TEST(Parse, Long) {
+ test_parse<long>();
+}
+
+TEST(Parse, ULongLong) {
+ test_parse<unsigned long long>();
+}
+
+TEST(Parse, LongLong) {
+ test_parse<long long>();
+}
+
+template<typename T>
+inline void test_consume() {
+ auto pos = "23"sv;
+ auto spacepos = " 23"sv;
+ auto neg = "-5"sv;
+ auto meow = "meow"sv;
+ auto trail = "9yards"sv;
+
+ auto v = pos;
+ auto r = consume<T>(v);
+ ASSERT_TRUE(r);
+ EXPECT_EQ(*r, 23);
+ EXPECT_TRUE(v.empty());
+
+ v = spacepos;
+ r = consume<T>(v);
+ EXPECT_FALSE(r);
+ EXPECT_EQ(v, spacepos);
+
+ v = neg;
+ r = consume<T>(v);
+ if constexpr (std::is_signed_v<T>) {
+ ASSERT_TRUE(r);
+ EXPECT_EQ(*r, -5);
+ EXPECT_TRUE(v.empty());
+ } else {
+ EXPECT_FALSE(r);
+ EXPECT_EQ(v, neg);
+ }
+
+ v = meow;
+ r = consume<T>(v);
+ EXPECT_FALSE(r);
+ EXPECT_EQ(v, meow);
+
+ v = trail;
+ r = consume<T>(v);
+ ASSERT_TRUE(r);
+ EXPECT_EQ(*r, 9);
+ auto w = trail;
+ w.remove_prefix(1);
+ EXPECT_EQ(v, w);
+}
+
+TEST(Consume, Char) {
+ test_consume<char>();
+}
+
+TEST(Consume, UChar) {
+ test_consume<unsigned char>();
+}
+
+TEST(Consume, SChar) {
+ test_consume<signed char>();
+}
+
+TEST(Consume, UInt8) {
+ test_consume<std::uint8_t>();
+}
+
+TEST(Consume, Int8) {
+ test_consume<std::int8_t>();
+}
+
+TEST(Consume, UInt16) {
+ test_consume<std::uint16_t>();
+}
+
+TEST(Consume, Int16) {
+ test_consume<std::int16_t>();
+}
+
+TEST(Consume, UInt32) {
+ test_consume<std::uint32_t>();
+}
+
+TEST(Consume, Int32) {
+ test_consume<std::int32_t>();
+}
+
+TEST(Consume, UInt64) {
+ test_consume<std::uint64_t>();
+}
+
+TEST(Consume, Int64) {
+ test_consume<std::int64_t>();
+}
+
+TEST(Consume, UIntMax) {
+ test_consume<std::uintmax_t>();
+}
+
+TEST(Consume, IntMax) {
+ test_consume<std::intmax_t>();
+}
+
+TEST(Consume, UIntPtr) {
+ test_consume<std::uintptr_t>();
+}
+
+TEST(Consume, IntPtr) {
+ test_consume<std::intptr_t>();
+}
+
+TEST(Consume, UShort) {
+ test_consume<unsigned short>();
+}
+
+TEST(Consume, Short) {
+ test_consume<short>();
+}
+
+TEST(Consume, ULong) {
+ test_consume<unsigned long>();
+}
+
+TEST(Consume, Long) {
+ test_consume<long>();
+}
+
+TEST(Consume, ULongLong) {
+ test_consume<unsigned long long>();
+}
+
+TEST(Consume, LongLong) {
+ test_consume<long long>();
+}
+
+
+
+/*
+ * Local Variables:
+ * compile-command: "cd .. ; make unittest_strtol && ./unittest_strtol"
+ * End:
+ */