mirror of
https://github.com/red0124/ssp.git
synced 2025-01-23 13:05:20 +01:00
266 lines
7.7 KiB
C++
266 lines
7.7 KiB
C++
#include "test_parser1.hpp"
|
|
|
|
struct my_string {
|
|
char* data{nullptr};
|
|
|
|
my_string() = default;
|
|
|
|
~my_string() {
|
|
delete[] data;
|
|
}
|
|
|
|
// make sure no object is copied
|
|
my_string(const my_string&) = delete;
|
|
my_string& operator=(const my_string&) = delete;
|
|
|
|
my_string(my_string&& other) : data{other.data} {
|
|
other.data = nullptr;
|
|
}
|
|
|
|
my_string& operator=(my_string&& other) {
|
|
data = other.data;
|
|
return *this;
|
|
}
|
|
};
|
|
|
|
template <>
|
|
inline bool ss::extract(const char* begin, const char* end, my_string& s) {
|
|
size_t size = end - begin;
|
|
s.data = new char[size + 1];
|
|
strncpy(s.data, begin, size);
|
|
s.data[size] = '\0';
|
|
return true;
|
|
}
|
|
|
|
struct xyz {
|
|
my_string x;
|
|
my_string y;
|
|
my_string z;
|
|
auto tied() {
|
|
return std::tie(x, y, z);
|
|
}
|
|
};
|
|
|
|
TEST_CASE_TEMPLATE("test moving of parsed composite values", T,
|
|
config<std::true_type>, config<std::false_type>,
|
|
config<std::true_type, ss::string_error>,
|
|
config<std::false_type, ss::string_error>) {
|
|
constexpr auto buffer_mode = T::BufferMode::value;
|
|
using ErrorMode = typename T::ErrorMode;
|
|
|
|
// to compile is enough
|
|
return;
|
|
auto [p, _] = make_parser<buffer_mode, ErrorMode>("", "");
|
|
std::ignore =
|
|
p.template try_next<my_string, my_string, my_string>()
|
|
.template or_else<my_string, my_string, my_string, my_string>(
|
|
[](auto&&) {})
|
|
.template or_else<my_string>([](auto&) {})
|
|
.template or_else<xyz>([](auto&&) {})
|
|
.template or_object<xyz, my_string, my_string, my_string>(
|
|
[](auto&&) {})
|
|
.template or_else<std::tuple<my_string, my_string, my_string>>(
|
|
[](auto&, auto&, auto&) {});
|
|
}
|
|
|
|
TEST_CASE_TEMPLATE("parser test string error mode", BufferMode, std::true_type,
|
|
std::false_type) {
|
|
unique_file_name f{"string_error"};
|
|
{
|
|
std::ofstream out{f.name};
|
|
out << "junk" << std::endl;
|
|
out << "junk" << std::endl;
|
|
}
|
|
|
|
auto [p, _] = make_parser<BufferMode::value, ss::string_error>(f.name, ",");
|
|
|
|
REQUIRE_FALSE(p.eof());
|
|
std::ignore = p.template get_next<int>();
|
|
CHECK_FALSE(p.valid());
|
|
CHECK_FALSE(p.error_msg().empty());
|
|
}
|
|
|
|
TEST_CASE_TEMPLATE("parser throw on error mode", BufferMode, std::true_type,
|
|
std::false_type) {
|
|
unique_file_name f{"throw_on_error"};
|
|
{
|
|
std::ofstream out{f.name};
|
|
out << "junk" << std::endl;
|
|
out << "junk" << std::endl;
|
|
}
|
|
|
|
auto [p, _] =
|
|
make_parser<BufferMode::value, ss::throw_on_error>(f.name, ",");
|
|
|
|
REQUIRE_FALSE(p.eof());
|
|
try {
|
|
std::ignore = p.template get_next<int>();
|
|
FAIL("Expected exception...");
|
|
} catch (const std::exception& e) {
|
|
CHECK_FALSE(std::string{e.what()}.empty());
|
|
}
|
|
}
|
|
|
|
static inline std::string no_quote(const std::string& s) {
|
|
if (!s.empty() && s[0] == '"') {
|
|
return {std::next(begin(s)), std::prev(end(s))};
|
|
}
|
|
return s;
|
|
}
|
|
|
|
TEST_CASE_TEMPLATE("test quote multiline", T, ParserOptionCombinations) {
|
|
constexpr auto buffer_mode = T::BufferMode::value;
|
|
using ErrorMode = typename T::ErrorMode;
|
|
|
|
unique_file_name f{"quote_multiline"};
|
|
std::vector<X> data = {{1, 2, "\"x\r\nx\nx\""},
|
|
{3, 4, "\"y\ny\r\ny\""},
|
|
{5, 6, "\"z\nz\""},
|
|
{7, 8, "\"u\"\"\""},
|
|
{9, 10, "v"},
|
|
{11, 12, "\"w\n\""}};
|
|
for (auto& [_, __, s] : data) {
|
|
update_if_crlf(s);
|
|
}
|
|
|
|
make_and_write(f.name, data);
|
|
for (auto& [_, __, s] : data) {
|
|
s = no_quote(s);
|
|
if (s[0] == 'u') {
|
|
s = "u\"";
|
|
}
|
|
}
|
|
|
|
auto [p, _] = make_parser<buffer_mode, ErrorMode, ss::multiline,
|
|
ss::quote<'"'>>(f.name, ",");
|
|
|
|
std::vector<X> i;
|
|
|
|
while (!p.eof()) {
|
|
auto a = p.template get_next<int, double, std::string>();
|
|
i.emplace_back(ss::to_object<X>(a));
|
|
}
|
|
|
|
for (auto& [_, __, s] : i) {
|
|
update_if_crlf(s);
|
|
}
|
|
CHECK_EQ(i, data);
|
|
|
|
auto [p_no_multiline, __] =
|
|
make_parser<buffer_mode, ErrorMode, ss::quote<'"'>>(f.name, ",");
|
|
while (!p.eof()) {
|
|
auto command = [&p_no_multiline = p_no_multiline] {
|
|
std::ignore =
|
|
p_no_multiline.template get_next<int, double, std::string>();
|
|
};
|
|
expect_error_on_command(p_no_multiline, command);
|
|
}
|
|
}
|
|
|
|
static inline std::string no_escape(std::string& s) {
|
|
s.erase(std::remove(begin(s), end(s), '\\'), end(s));
|
|
return s;
|
|
}
|
|
|
|
TEST_CASE_TEMPLATE("test escape multiline", T, ParserOptionCombinations) {
|
|
constexpr auto buffer_mode = T::BufferMode::value;
|
|
using ErrorMode = typename T::ErrorMode;
|
|
|
|
unique_file_name f{"escape_multiline"};
|
|
std::vector<X> data = {{1, 2, "x\\\nx\\\r\nx"},
|
|
{5, 6, "z\\\nz\\\nz"},
|
|
{7, 8, "u"},
|
|
{3, 4, "y\\\ny\\\ny"},
|
|
{9, 10, "v\\\\"},
|
|
{11, 12, "w\\\n"}};
|
|
for (auto& [_, __, s] : data) {
|
|
update_if_crlf(s);
|
|
}
|
|
|
|
make_and_write(f.name, data);
|
|
for (auto& [_, __, s] : data) {
|
|
s = no_escape(s);
|
|
if (s == "v") {
|
|
s = "v\\";
|
|
}
|
|
}
|
|
|
|
auto [p, _] = make_parser<buffer_mode, ErrorMode, ss::multiline,
|
|
ss::escape<'\\'>>(f.name, ",");
|
|
std::vector<X> i;
|
|
|
|
while (!p.eof()) {
|
|
auto a = p.template get_next<int, double, std::string>();
|
|
i.emplace_back(ss::to_object<X>(a));
|
|
}
|
|
|
|
for (auto& [_, __, s] : i) {
|
|
update_if_crlf(s);
|
|
}
|
|
CHECK_EQ(i, data);
|
|
|
|
auto [p_no_multiline, __] =
|
|
make_parser<buffer_mode, ErrorMode, ss::escape<'\\'>>(f.name, ",");
|
|
while (!p.eof()) {
|
|
auto command = [&p_no_multiline = p_no_multiline] {
|
|
auto a =
|
|
p_no_multiline.template get_next<int, double, std::string>();
|
|
};
|
|
expect_error_on_command(p_no_multiline, command);
|
|
}
|
|
}
|
|
|
|
TEST_CASE_TEMPLATE("test quote escape multiline", T, ParserOptionCombinations) {
|
|
constexpr auto buffer_mode = T::BufferMode::value;
|
|
using ErrorMode = typename T::ErrorMode;
|
|
|
|
unique_file_name f{"quote_escape_multiline"};
|
|
{
|
|
std::ofstream out{f.name};
|
|
out << "1,2,\"just\\\n\nstrings\"" << std::endl;
|
|
#ifndef _WIN32
|
|
out << "3,4,\"just\r\nsome\\\r\n\n\\\nstrings\"" << std::endl;
|
|
out << "5,6,\"just\\\n\\\r\n\r\n\nstrings" << std::endl;
|
|
#else
|
|
out << "3,4,\"just\nsome\\\n\n\\\nstrings\"" << std::endl;
|
|
out << "5,6,\"just\\\n\\\n\n\nstrings" << std::endl;
|
|
#endif
|
|
out << "7,8,\"just strings\"" << std::endl;
|
|
out << "9,10,just strings" << std::endl;
|
|
}
|
|
size_t bad_lines = 1;
|
|
auto num_errors = 0;
|
|
|
|
auto [p, _] = make_parser<buffer_mode, ErrorMode, ss::multiline,
|
|
ss::escape<'\\'>, ss::quote<'"'>>(f.name);
|
|
std::vector<X> i;
|
|
|
|
while (!p.eof()) {
|
|
try {
|
|
auto a = p.template get_next<int, double, std::string>();
|
|
if (p.valid()) {
|
|
i.emplace_back(ss::to_object<X>(a));
|
|
} else {
|
|
++num_errors;
|
|
}
|
|
} catch (const std::exception& e) {
|
|
++num_errors;
|
|
}
|
|
}
|
|
|
|
CHECK(bad_lines == num_errors);
|
|
|
|
std::vector<X> data = {{1, 2, "just\n\nstrings"},
|
|
#ifndef _WIN32
|
|
{3, 4, "just\r\nsome\r\n\n\nstrings"},
|
|
#else
|
|
{3, 4, "just\nsome\n\n\nstrings"},
|
|
#endif
|
|
{9, 10, "just strings"}};
|
|
|
|
for (auto& [_, __, s] : i) {
|
|
update_if_crlf(s);
|
|
}
|
|
CHECK_EQ(i, data);
|
|
}
|