draconisplusplus/include/rfl/parsing/Parser_wstring.hpp

82 lines
2.3 KiB
C++
Raw Normal View History

2024-05-31 22:59:00 -04:00
#ifndef RFL_PARSING_PARSER_WSTRING_HPP_
#define RFL_PARSING_PARSER_WSTRING_HPP_
#include <map>
#include <type_traits>
#include "../Result.hpp"
#include "../always_false.hpp"
#include "Parent.hpp"
#include "Parser_base.hpp"
#include "schema/Type.hpp"
namespace rfl {
2024-06-08 14:10:59 -04:00
namespace parsing {
2024-05-31 22:59:00 -04:00
2024-06-08 14:10:59 -04:00
template <class R, class W, class ProcessorsType>
requires AreReaderAndWriter<R, W, std::wstring>
struct Parser<R, W, std::wstring, ProcessorsType> {
public:
using InputVarType = typename R::InputVarType;
using OutputVarType = typename W::OutputVarType;
2024-05-31 22:59:00 -04:00
2024-06-08 14:10:59 -04:00
using ParentType = Parent<W>;
2024-05-31 22:59:00 -04:00
2024-06-08 15:53:06 -04:00
static Result<std::wstring>
read(const R& _r, const InputVarType& _var) noexcept {
2024-06-08 14:10:59 -04:00
if (_r.is_empty(_var)) { return std::wstring(); }
2024-05-31 22:59:00 -04:00
2024-06-08 14:10:59 -04:00
auto inStr = Parser<R, W, std::string, ProcessorsType>::read(_r, _var);
if (auto err = inStr.error(); err.has_value()) {
return Result<std::wstring>(err.value());
}
2024-05-31 22:59:00 -04:00
2024-06-08 14:10:59 -04:00
std::mbstate_t state = std::mbstate_t();
2024-06-08 15:53:06 -04:00
auto val = inStr.value();
2024-05-31 22:59:00 -04:00
2024-06-08 14:10:59 -04:00
std::wstring outStr(val.size() * 2, L'\0');
2024-05-31 22:59:00 -04:00
2024-06-08 14:10:59 -04:00
// Explicitly set the size so we don't empty it when we truncate
outStr.resize(val.size() * 2);
2024-05-31 22:59:00 -04:00
2024-06-08 14:10:59 -04:00
auto* ptr = val.c_str();
2024-05-31 22:59:00 -04:00
2024-06-08 14:10:59 -04:00
// Add 1 for null terminator
auto len = std::mbsrtowcs(outStr.data(), &ptr, val.size(), &state);
outStr.resize(len); // Truncate the extra bytes
2024-05-31 22:59:00 -04:00
2024-06-08 14:10:59 -04:00
return Result<std::wstring>(outStr);
}
2024-05-31 22:59:00 -04:00
2024-06-08 14:10:59 -04:00
template <class P>
2024-06-08 15:53:06 -04:00
static void
write(const W& _w, const std::wstring& _str, const P& _parent) noexcept {
2024-06-08 14:10:59 -04:00
if (_str.empty()) {
ParentType::add_value(_w, std::string(), _parent);
return;
}
2024-05-31 22:59:00 -04:00
2024-06-08 14:10:59 -04:00
std::mbstate_t state = std::mbstate_t();
2024-06-08 15:53:06 -04:00
std::string outStr(_str.size(), '\0');
2024-06-08 14:10:59 -04:00
outStr.resize(_str.size());
2024-05-31 22:59:00 -04:00
2024-06-08 14:10:59 -04:00
auto* ptr = _str.c_str();
2024-06-08 15:53:06 -04:00
auto len = std::wcsrtombs(outStr.data(), &ptr, _str.size(), &state);
2024-06-08 14:10:59 -04:00
outStr.resize(len);
2024-05-31 22:59:00 -04:00
2024-06-08 14:10:59 -04:00
ParentType::add_value(_w, outStr, _parent);
}
2024-05-31 22:59:00 -04:00
2024-06-08 14:10:59 -04:00
static schema::Type to_schema(
2024-06-08 15:53:06 -04:00
std::map<std::string, schema::Type>* _definitions
) {
return Parser<R, W, std::string, ProcessorsType>::to_schema(_definitions
);
2024-06-08 14:10:59 -04:00
}
};
2024-05-31 22:59:00 -04:00
2024-06-08 14:10:59 -04:00
} // namespace parsing
} // namespace rfl
2024-05-31 22:59:00 -04:00
#endif