2021-02-14 01:59:06 +01:00
|
|
|
#pragma once
|
2024-02-22 23:59:58 +01:00
|
|
|
#include <cerrno>
|
2021-02-28 19:22:54 +01:00
|
|
|
#include <cstdio>
|
|
|
|
#include <cstdlib>
|
2024-02-22 23:59:58 +01:00
|
|
|
#include <cstring>
|
2021-02-14 01:59:06 +01:00
|
|
|
#include <vector>
|
|
|
|
|
2024-03-12 18:31:24 +01:00
|
|
|
#if !__unix__
|
|
|
|
#include <array>
|
|
|
|
#include <cstdint>
|
|
|
|
#endif
|
|
|
|
|
2021-02-14 01:59:06 +01:00
|
|
|
namespace ss {
|
|
|
|
|
|
|
|
struct none {};
|
|
|
|
|
|
|
|
using string_range = std::pair<const char*, const char*>;
|
|
|
|
using split_data = std::vector<string_range>;
|
|
|
|
|
|
|
|
constexpr inline auto default_delimiter = ",";
|
2024-02-22 00:30:05 +01:00
|
|
|
constexpr inline auto get_line_initial_buffer_size = 128;
|
2021-02-14 01:59:06 +01:00
|
|
|
|
|
|
|
template <bool StringError>
|
2024-03-12 18:31:24 +01:00
|
|
|
void assert_string_error_defined() {
|
2021-02-14 01:59:06 +01:00
|
|
|
static_assert(StringError,
|
|
|
|
"'string_error' needs to be enabled to use 'error_msg'");
|
|
|
|
}
|
|
|
|
|
2023-08-04 16:48:07 +02:00
|
|
|
template <bool ThrowOnError>
|
2024-03-12 18:31:24 +01:00
|
|
|
void assert_throw_on_error_not_defined() {
|
2023-08-04 16:48:07 +02:00
|
|
|
static_assert(!ThrowOnError, "cannot handle errors manually if "
|
|
|
|
"'throw_on_error' is enabled");
|
|
|
|
}
|
|
|
|
|
2024-02-25 10:42:11 +01:00
|
|
|
inline void* strict_realloc(void* ptr, size_t size) {
|
2024-02-25 13:02:34 +01:00
|
|
|
ptr = std::realloc(ptr, size);
|
2024-02-25 10:42:11 +01:00
|
|
|
if (!ptr) {
|
|
|
|
throw std::bad_alloc{};
|
|
|
|
}
|
|
|
|
|
|
|
|
return ptr;
|
|
|
|
}
|
|
|
|
|
2021-02-21 22:25:58 +01:00
|
|
|
#if __unix__
|
2024-03-01 02:47:04 +01:00
|
|
|
inline ssize_t get_line_file(char*& lineptr, size_t& n, FILE* file) {
|
|
|
|
return getline(&lineptr, &n, file);
|
2021-02-21 22:25:58 +01:00
|
|
|
}
|
|
|
|
#else
|
2024-02-22 00:30:05 +01:00
|
|
|
|
2024-02-24 19:09:41 +01:00
|
|
|
using ssize_t = intptr_t;
|
2021-02-21 22:22:18 +01:00
|
|
|
|
2024-03-12 10:22:10 +01:00
|
|
|
inline ssize_t get_line_file(char*& lineptr, size_t& n, FILE* file) {
|
2024-03-12 18:31:24 +01:00
|
|
|
std::array<char, get_line_initial_buffer_size> buff;
|
2021-02-21 22:22:18 +01:00
|
|
|
|
2024-03-01 02:47:04 +01:00
|
|
|
if (lineptr == nullptr || n < sizeof(buff)) {
|
2024-02-23 00:59:58 +01:00
|
|
|
size_t new_n = sizeof(buff);
|
2024-03-01 02:47:04 +01:00
|
|
|
lineptr = static_cast<char*>(strict_realloc(lineptr, new_n));
|
|
|
|
n = new_n;
|
2021-02-21 22:22:18 +01:00
|
|
|
}
|
|
|
|
|
2024-03-01 02:47:04 +01:00
|
|
|
lineptr[0] = '\0';
|
2024-02-22 23:59:58 +01:00
|
|
|
|
2024-02-23 23:25:03 +01:00
|
|
|
size_t line_used = 0;
|
2024-03-12 18:31:24 +01:00
|
|
|
while (std::fgets(buff.data(), sizeof(buff), file) != nullptr) {
|
2024-03-01 02:47:04 +01:00
|
|
|
line_used = std::strlen(lineptr);
|
2024-03-12 18:31:24 +01:00
|
|
|
size_t buff_used = std::strlen(buff.data());
|
2024-02-22 23:59:58 +01:00
|
|
|
|
2024-03-01 02:47:04 +01:00
|
|
|
if (n <= buff_used + line_used) {
|
|
|
|
size_t new_n = n * 2;
|
|
|
|
lineptr = static_cast<char*>(strict_realloc(lineptr, new_n));
|
|
|
|
n = new_n;
|
2021-02-21 22:22:18 +01:00
|
|
|
}
|
|
|
|
|
2024-03-12 18:31:24 +01:00
|
|
|
std::memcpy(lineptr + line_used, buff.data(), buff_used);
|
2024-02-22 23:59:58 +01:00
|
|
|
line_used += buff_used;
|
2024-03-01 02:47:04 +01:00
|
|
|
lineptr[line_used] = '\0';
|
2024-02-22 23:59:58 +01:00
|
|
|
|
2024-03-01 02:47:04 +01:00
|
|
|
if (lineptr[line_used - 1] == '\n') {
|
2024-02-22 23:59:58 +01:00
|
|
|
return line_used;
|
2021-02-21 22:22:18 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-02-23 23:25:03 +01:00
|
|
|
return (line_used != 0) ? line_used : -1;
|
2021-02-21 22:22:18 +01:00
|
|
|
}
|
2024-02-22 23:59:58 +01:00
|
|
|
|
2021-02-21 22:22:18 +01:00
|
|
|
#endif
|
|
|
|
|
2024-03-12 10:22:10 +01:00
|
|
|
inline ssize_t get_line_buffer(char*& lineptr, size_t& n,
|
2024-03-01 02:47:04 +01:00
|
|
|
const char* const csv_data_buffer, size_t csv_data_size,
|
|
|
|
size_t& curr_char) {
|
|
|
|
if (curr_char >= csv_data_size) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (lineptr == nullptr || n < get_line_initial_buffer_size) {
|
2024-03-12 18:31:24 +01:00
|
|
|
auto* new_lineptr = static_cast<char*>(
|
2024-03-01 02:47:04 +01:00
|
|
|
strict_realloc(lineptr, get_line_initial_buffer_size));
|
|
|
|
lineptr = new_lineptr;
|
|
|
|
n = get_line_initial_buffer_size;
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t line_used = 0;
|
|
|
|
while (curr_char < csv_data_size) {
|
|
|
|
if (line_used + 1 >= n) {
|
|
|
|
size_t new_n = n * 2;
|
|
|
|
|
|
|
|
char* new_lineptr =
|
|
|
|
static_cast<char*>(strict_realloc(lineptr, new_n));
|
|
|
|
n = new_n;
|
|
|
|
lineptr = new_lineptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto c = csv_data_buffer[curr_char++];
|
|
|
|
lineptr[line_used++] = c;
|
|
|
|
if (c == '\n') {
|
|
|
|
lineptr[line_used] = '\0';
|
|
|
|
return line_used;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-03-12 10:22:10 +01:00
|
|
|
lineptr[line_used] = '\0';
|
|
|
|
return line_used;
|
2024-03-01 02:47:04 +01:00
|
|
|
}
|
|
|
|
|
2024-03-12 10:22:10 +01:00
|
|
|
inline std::tuple<ssize_t, bool> get_line(char*& buffer, size_t& buffer_size,
|
2024-03-01 02:47:04 +01:00
|
|
|
FILE* file,
|
|
|
|
const char* const csv_data_buffer,
|
|
|
|
size_t csv_data_size, size_t& curr_char) {
|
2024-03-12 18:31:24 +01:00
|
|
|
ssize_t ssize = 0;
|
2024-03-01 02:47:04 +01:00
|
|
|
if (file) {
|
|
|
|
ssize = get_line_file(buffer, buffer_size, file);
|
2024-03-02 00:34:19 +01:00
|
|
|
curr_char += ssize;
|
2024-03-01 02:47:04 +01:00
|
|
|
} else {
|
|
|
|
ssize = get_line_buffer(buffer, buffer_size, csv_data_buffer,
|
|
|
|
csv_data_size, curr_char);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ssize == -1) {
|
|
|
|
if (errno == ENOMEM) {
|
|
|
|
throw std::bad_alloc{};
|
|
|
|
}
|
|
|
|
return {ssize, true};
|
|
|
|
}
|
|
|
|
|
|
|
|
return {ssize, false};
|
|
|
|
}
|
|
|
|
|
2024-03-12 10:22:10 +01:00
|
|
|
} /* namespace ss */
|