2019-06-08 16:50:03 +03:00
|
|
|
// Formatting library for C++ - scanning API test
|
2019-05-15 20:02:40 +03:00
|
|
|
//
|
2019-06-08 16:50:03 +03:00
|
|
|
// Copyright (c) 2019 - present, Victor Zverovich
|
2019-05-15 20:02:40 +03:00
|
|
|
// All rights reserved.
|
|
|
|
//
|
|
|
|
// For the license information refer to format.h.
|
|
|
|
|
2020-05-08 01:59:46 +03:00
|
|
|
#include "scan.h"
|
|
|
|
|
2019-06-08 16:50:03 +03:00
|
|
|
#include <time.h>
|
2020-05-08 01:59:46 +03:00
|
|
|
|
2019-05-15 20:20:51 +03:00
|
|
|
#include <climits>
|
2023-12-25 18:18:23 +03:00
|
|
|
#include <thread>
|
2019-05-15 20:02:40 +03:00
|
|
|
|
2023-12-02 20:34:27 +03:00
|
|
|
#include "fmt/os.h"
|
2021-04-29 01:59:43 +03:00
|
|
|
#include "gmock/gmock.h"
|
2019-05-15 21:05:19 +03:00
|
|
|
#include "gtest-extra.h"
|
2019-05-15 20:02:40 +03:00
|
|
|
|
2021-05-06 03:43:06 +03:00
|
|
|
TEST(scan_test, read_text) {
|
2023-12-02 20:34:27 +03:00
|
|
|
fmt::string_view s = "foo";
|
2019-05-19 17:02:41 +03:00
|
|
|
auto end = fmt::scan(s, "foo");
|
|
|
|
EXPECT_EQ(end, s.end());
|
|
|
|
EXPECT_THROW_MSG(fmt::scan("fob", "foo"), fmt::format_error, "invalid input");
|
|
|
|
}
|
|
|
|
|
2021-05-06 03:43:06 +03:00
|
|
|
TEST(scan_test, read_int) {
|
2023-12-02 20:34:27 +03:00
|
|
|
int n = 0;
|
2019-05-15 20:02:40 +03:00
|
|
|
fmt::scan("42", "{}", n);
|
|
|
|
EXPECT_EQ(n, 42);
|
2019-05-19 17:02:41 +03:00
|
|
|
fmt::scan("-42", "{}", n);
|
|
|
|
EXPECT_EQ(n, -42);
|
2023-12-22 21:50:01 +03:00
|
|
|
EXPECT_THROW_MSG(fmt::scan(std::to_string(INT_MAX + 1u), "{}", n),
|
|
|
|
fmt::format_error, "number is too big");
|
2019-05-19 17:02:41 +03:00
|
|
|
}
|
|
|
|
|
2021-05-06 03:43:06 +03:00
|
|
|
TEST(scan_test, read_longlong) {
|
2019-05-19 17:02:41 +03:00
|
|
|
long long n = 0;
|
|
|
|
fmt::scan("42", "{}", n);
|
|
|
|
EXPECT_EQ(n, 42);
|
|
|
|
fmt::scan("-42", "{}", n);
|
|
|
|
EXPECT_EQ(n, -42);
|
|
|
|
}
|
|
|
|
|
2021-05-06 03:43:06 +03:00
|
|
|
TEST(scan_test, read_uint) {
|
2023-12-02 20:34:27 +03:00
|
|
|
unsigned n = 0;
|
2019-05-19 17:02:41 +03:00
|
|
|
fmt::scan("42", "{}", n);
|
|
|
|
EXPECT_EQ(n, 42);
|
|
|
|
EXPECT_THROW_MSG(fmt::scan("-42", "{}", n), fmt::format_error,
|
|
|
|
"invalid input");
|
|
|
|
}
|
|
|
|
|
2021-05-06 03:43:06 +03:00
|
|
|
TEST(scan_test, read_ulonglong) {
|
2019-05-19 17:02:41 +03:00
|
|
|
unsigned long long n = 0;
|
|
|
|
fmt::scan("42", "{}", n);
|
|
|
|
EXPECT_EQ(n, 42);
|
|
|
|
EXPECT_THROW_MSG(fmt::scan("-42", "{}", n), fmt::format_error,
|
|
|
|
"invalid input");
|
2019-05-15 20:02:40 +03:00
|
|
|
}
|
2019-06-02 21:11:28 +03:00
|
|
|
|
2021-05-06 03:43:06 +03:00
|
|
|
TEST(scan_test, read_string) {
|
2023-12-02 20:34:27 +03:00
|
|
|
std::string s;
|
2019-06-02 17:30:26 +03:00
|
|
|
fmt::scan("foo", "{}", s);
|
|
|
|
EXPECT_EQ(s, "foo");
|
|
|
|
}
|
2019-05-15 21:05:19 +03:00
|
|
|
|
2021-05-06 03:43:06 +03:00
|
|
|
TEST(scan_test, read_string_view) {
|
2023-12-02 20:34:27 +03:00
|
|
|
fmt::string_view s;
|
2019-06-02 21:11:28 +03:00
|
|
|
fmt::scan("foo", "{}", s);
|
|
|
|
EXPECT_EQ(s, "foo");
|
|
|
|
}
|
|
|
|
|
2023-12-23 17:53:25 +03:00
|
|
|
TEST(scan_test, separator) {
|
|
|
|
int n1 = 0, n2 = 0;
|
|
|
|
fmt::scan("10 20", "{} {}", n1, n2);
|
|
|
|
EXPECT_EQ(n1, 10);
|
|
|
|
EXPECT_EQ(n2, 20);
|
|
|
|
}
|
|
|
|
|
2022-11-26 14:42:02 +03:00
|
|
|
#ifdef FMT_HAVE_STRPTIME
|
2023-12-23 17:53:25 +03:00
|
|
|
struct num {
|
|
|
|
int value;
|
|
|
|
};
|
|
|
|
|
2019-06-08 16:50:03 +03:00
|
|
|
namespace fmt {
|
2023-12-23 17:53:25 +03:00
|
|
|
template <> struct scanner<num> {
|
|
|
|
bool hex = false;
|
2019-06-08 16:50:03 +03:00
|
|
|
|
2023-11-25 18:41:04 +03:00
|
|
|
auto parse(scan_parse_context& ctx) -> scan_parse_context::iterator {
|
2023-12-23 17:53:25 +03:00
|
|
|
auto it = ctx.begin(), end = ctx.end();
|
|
|
|
if (it != end && *it == 'x') hex = true;
|
|
|
|
if (it != end && *it != '}') throw_format_error("invalid format");
|
|
|
|
return it;
|
2019-06-08 16:50:03 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
template <class ScanContext>
|
2023-12-23 17:53:25 +03:00
|
|
|
auto scan(num&, ScanContext& ctx) const -> typename ScanContext::iterator {
|
|
|
|
// TODO
|
|
|
|
//return fmt::scan({ctx.begin(), ctx.end()}, "{}", n.value);
|
2023-12-02 20:34:27 +03:00
|
|
|
return ctx.begin();
|
2019-06-08 16:50:03 +03:00
|
|
|
}
|
|
|
|
};
|
|
|
|
} // namespace fmt
|
|
|
|
|
2021-05-06 03:43:06 +03:00
|
|
|
TEST(scan_test, read_custom) {
|
2023-12-23 17:53:25 +03:00
|
|
|
auto input = "42";
|
|
|
|
auto n = num();
|
|
|
|
fmt::scan(input, "{:}", n);
|
|
|
|
//EXPECT_EQ(n, 42);
|
2019-06-08 16:50:03 +03:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2021-05-06 03:43:06 +03:00
|
|
|
TEST(scan_test, invalid_format) {
|
2019-05-15 21:05:19 +03:00
|
|
|
EXPECT_THROW_MSG(fmt::scan("", "{}"), fmt::format_error,
|
|
|
|
"argument index out of range");
|
|
|
|
EXPECT_THROW_MSG(fmt::scan("", "{"), fmt::format_error,
|
|
|
|
"invalid format string");
|
|
|
|
}
|
2019-06-02 17:30:26 +03:00
|
|
|
|
2021-05-06 03:43:06 +03:00
|
|
|
TEST(scan_test, example) {
|
2023-12-23 17:53:25 +03:00
|
|
|
std::string key;
|
|
|
|
int value = 0;
|
2019-06-02 17:30:26 +03:00
|
|
|
fmt::scan("answer = 42", "{} = {}", key, value);
|
|
|
|
EXPECT_EQ(key, "answer");
|
|
|
|
EXPECT_EQ(value, 42);
|
|
|
|
}
|
2023-11-26 20:22:31 +03:00
|
|
|
|
2023-12-25 18:18:23 +03:00
|
|
|
TEST(scan_test, end_of_input) {
|
|
|
|
int value = 0;
|
|
|
|
fmt::scan("", "{}", value);
|
|
|
|
}
|
|
|
|
|
2023-11-26 20:22:31 +03:00
|
|
|
#if FMT_USE_FCNTL
|
|
|
|
TEST(scan_test, file) {
|
|
|
|
fmt::file read_end, write_end;
|
|
|
|
fmt::file::pipe(read_end, write_end);
|
2023-12-24 18:32:27 +03:00
|
|
|
|
|
|
|
fmt::string_view input = "10 20";
|
2023-11-26 20:22:31 +03:00
|
|
|
write_end.write(input.data(), input.size());
|
|
|
|
write_end.close();
|
2023-12-24 18:32:27 +03:00
|
|
|
|
|
|
|
int n1 = 0, n2 = 0;
|
|
|
|
fmt::buffered_file f = read_end.fdopen("r");
|
|
|
|
fmt::scan(f.get(), "{} {}", n1, n2);
|
|
|
|
EXPECT_EQ(n1, 10);
|
|
|
|
EXPECT_EQ(n2, 20);
|
2023-11-26 20:22:31 +03:00
|
|
|
}
|
2023-12-25 19:39:14 +03:00
|
|
|
|
|
|
|
TEST(scan_test, lock) {
|
|
|
|
fmt::file read_end, write_end;
|
|
|
|
fmt::file::pipe(read_end, write_end);
|
|
|
|
|
|
|
|
std::thread producer([&]() {
|
|
|
|
fmt::string_view input = "42 ";
|
|
|
|
for (int i = 0; i < 1000; ++i)
|
|
|
|
write_end.write(input.data(), input.size());
|
|
|
|
write_end.close();
|
|
|
|
});
|
|
|
|
|
|
|
|
fmt::buffered_file f = read_end.fdopen("r");
|
|
|
|
auto fun = [&]() {
|
|
|
|
int value = 0;
|
|
|
|
while (fmt::scan(f.get(), "{}", value)) {
|
|
|
|
if (value != 42) {
|
|
|
|
read_end.close();
|
|
|
|
EXPECT_EQ(value, 42);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
std::thread consumer1(fun);
|
|
|
|
std::thread consumer2(fun);
|
|
|
|
producer.join();
|
|
|
|
consumer1.join();
|
|
|
|
consumer2.join();
|
|
|
|
}
|
2023-11-26 20:22:31 +03:00
|
|
|
#endif // FMT_USE_FCNTL
|