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 <cstdint>
|
|
|
|
#include <cstdio>
|
|
|
|
#include <cstdlib>
|
2024-02-22 23:59:58 +01:00
|
|
|
#include <cstring>
|
2021-02-14 01:59:06 +01:00
|
|
|
#include <vector>
|
|
|
|
|
|
|
|
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 23:59:58 +01:00
|
|
|
constexpr inline auto get_line_initial_buffer_size = 128;
|
2021-02-14 01:59:06 +01:00
|
|
|
|
|
|
|
template <bool StringError>
|
|
|
|
inline void assert_string_error_defined() {
|
|
|
|
static_assert(StringError,
|
|
|
|
"'string_error' needs to be enabled to use 'error_msg'");
|
|
|
|
}
|
|
|
|
|
2023-08-04 16:48:07 +02:00
|
|
|
template <bool ThrowOnError>
|
|
|
|
inline void assert_throw_on_error_not_defined() {
|
|
|
|
static_assert(!ThrowOnError, "cannot handle errors manually if "
|
|
|
|
"'throw_on_error' is enabled");
|
|
|
|
}
|
|
|
|
|
2024-02-23 00:59:58 +01:00
|
|
|
#if __unix__XX
|
2021-02-21 22:25:58 +01:00
|
|
|
inline ssize_t get_line(char** lineptr, size_t* n, FILE* stream) {
|
|
|
|
return getline(lineptr, n, stream);
|
|
|
|
}
|
|
|
|
#else
|
2024-02-22 23:59:58 +01:00
|
|
|
|
2021-02-21 22:22:18 +01:00
|
|
|
using ssize_t = int64_t;
|
|
|
|
|
2024-02-22 23:59:58 +01:00
|
|
|
ssize_t get_line(char** lineptr, size_t* n, FILE* fp) {
|
|
|
|
if (lineptr == nullptr || n == nullptr || fp == nullptr) {
|
2021-02-21 22:22:18 +01:00
|
|
|
errno = EINVAL;
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2024-02-22 23:59:58 +01:00
|
|
|
char buff[get_line_initial_buffer_size];
|
2021-02-21 22:22:18 +01:00
|
|
|
|
2024-02-22 23:59:58 +01:00
|
|
|
if (*lineptr == nullptr || *n < sizeof(buff)) {
|
2024-02-23 00:59:58 +01:00
|
|
|
size_t new_n = sizeof(buff);
|
|
|
|
auto new_lineptr = static_cast<char*>(realloc(*lineptr, new_n));
|
2024-02-22 23:59:58 +01:00
|
|
|
if (new_lineptr == nullptr) {
|
|
|
|
errno = ENOMEM;
|
2021-02-21 22:22:18 +01:00
|
|
|
return -1;
|
|
|
|
}
|
2024-02-22 23:59:58 +01:00
|
|
|
|
|
|
|
*lineptr = new_lineptr;
|
2024-02-23 00:59:58 +01:00
|
|
|
*n = new_n;
|
2021-02-21 22:22:18 +01:00
|
|
|
}
|
|
|
|
|
2024-02-22 23:59:58 +01:00
|
|
|
(*lineptr)[0] = '\0';
|
|
|
|
|
|
|
|
while (fgets(buff, sizeof(buff), fp) != nullptr) {
|
|
|
|
size_t line_used = strlen(*lineptr);
|
|
|
|
size_t buff_used = strlen(buff);
|
|
|
|
|
|
|
|
if (*n < buff_used + line_used) {
|
2024-02-23 00:59:58 +01:00
|
|
|
size_t new_n = *n * 2;
|
2024-02-22 23:59:58 +01:00
|
|
|
|
|
|
|
auto new_lineptr = static_cast<char*>(realloc(*lineptr, *n));
|
|
|
|
if (new_lineptr == nullptr) {
|
|
|
|
errno = ENOMEM;
|
2021-02-21 22:22:18 +01:00
|
|
|
return -1;
|
|
|
|
}
|
2024-02-22 23:59:58 +01:00
|
|
|
|
|
|
|
*lineptr = new_lineptr;
|
2024-02-23 00:59:58 +01:00
|
|
|
*n = new_n;
|
2021-02-21 22:22:18 +01:00
|
|
|
}
|
|
|
|
|
2024-02-22 23:59:58 +01:00
|
|
|
memcpy(*lineptr + line_used, buff, buff_used);
|
|
|
|
line_used += buff_used;
|
|
|
|
(*lineptr)[line_used] = '\0';
|
|
|
|
|
|
|
|
if ((*lineptr)[line_used - 1] == '\n') {
|
|
|
|
return line_used;
|
2021-02-21 22:22:18 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-02-22 23:59:58 +01:00
|
|
|
return -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
|
|
|
|
|
2021-02-14 01:59:06 +01:00
|
|
|
} /* ss */
|