// This file Copyright © 2008-2022 Mnemosyne LLC. // It may be used under GPLv2 (SPDX: GPL-2.0), GPLv3 (SPDX: GPL-3.0), // or any future license endorsed by Mnemosyne LLC. // License text can be found in the licenses/ folder. #include #include /* isdigit() */ #include #include #include #include #include #define LIBTRANSMISSION_VARIANT_MODULE #include "transmission.h" #include "benc.h" #include "tr-assert.h" #include "quark.h" #include "utils.h" /* tr_snprintf() */ #include "variant-common.h" #include "variant.h" using namespace std::literals; auto constexpr MaxBencStrLength = size_t{ 128 * 1024 * 1024 }; // arbitrary /*** **** **** ***/ namespace transmission::benc::impl { /** * The initial i and trailing e are beginning and ending delimiters. * You can have negative numbers such as i-3e. You cannot prefix the * number with a zero such as i04e. However, i0e is valid. * Example: i3e represents the integer "3" * * The maximum number of bit of this integer is unspecified, * but to handle it as a signed 64bit integer is mandatory to handle * "large files" aka .torrent for more that 4Gbyte */ std::optional ParseInt(std::string_view* benc) { auto constexpr Prefix = "i"sv; auto constexpr Suffix = "e"sv; // find the beginning delimiter auto walk = *benc; if (std::size(walk) < 3 || !tr_strvStartsWith(walk, Prefix)) { return {}; } // find the ending delimiter walk.remove_prefix(std::size(Prefix)); auto const pos = walk.find(Suffix); if (pos == std::string_view::npos) { return {}; } // leading zeroes are not allowed if ((walk[0] == '0' && isdigit(walk[1])) || (walk[0] == '-' && walk[1] == '0' && isdigit(walk[2]))) { return {}; } // parse the string and make sure the next char is `Suffix` auto const value = tr_parseNum(walk); if (!value || !tr_strvStartsWith(walk, Suffix)) { return {}; } walk.remove_prefix(std::size(Suffix)); *benc = walk; return *value; } /** * Byte strings are encoded as follows: * : * Note that there is no constant beginning delimiter, and no ending delimiter. * Example: 4:spam represents the string "spam" */ std::optional ParseString(std::string_view* benc) { // find the ':' delimiter auto const colon_pos = benc->find(':'); if (colon_pos == std::string_view::npos) { return {}; } // get the string length auto svtmp = benc->substr(0, colon_pos); auto const len = tr_parseNum(svtmp); if (!len || *len >= MaxBencStrLength) { return {}; } // do we have `len` bytes of string data? svtmp = benc->substr(colon_pos + 1); if (std::size(svtmp) < len) { return {}; } auto const string = svtmp.substr(0, *len); *benc = svtmp.substr(*len); return string; } } // namespace transmission::benc::impl /*** **** tr_variantParse() **** tr_variantLoad() ***/ struct MyHandler : public transmission::benc::Handler { tr_variant* const top_; int const parse_opts_; std::deque stack_; std::optional key_; MyHandler(tr_variant* top, int parse_opts) : top_{ top } , parse_opts_{ parse_opts } { } virtual ~MyHandler() = default; bool Int64(int64_t value) final { if (tr_variant* variant = get_node(); variant != nullptr) { tr_variantInitInt(variant, value); } return true; } bool String(std::string_view sv) final { if (tr_variant* variant = get_node(); variant != nullptr) { if ((parse_opts_ & TR_VARIANT_PARSE_INPLACE) != 0) { tr_variantInitStrView(variant, sv); } else { tr_variantInitStr(variant, sv); } } return true; } bool StartDict() final { if (tr_variant* variant = get_node(); variant != nullptr) { tr_variantInitDict(variant, 0); stack_.push_back(variant); } return true; } bool Key(std::string_view sv) final { key_ = tr_quark_new(sv); return true; } bool EndDict() final { stack_.pop_back(); return true; } bool StartArray() final { if (tr_variant* variant = get_node(); variant != nullptr) { tr_variantInitList(variant, 0); stack_.push_back(variant); } return true; } bool EndArray() final { stack_.pop_back(); return true; } private: tr_variant* get_node() { tr_variant* node = nullptr; if (std::empty(stack_)) { node = top_; } else { auto* parent = stack_.back(); if (tr_variantIsList(parent)) { node = tr_variantListAdd(parent); } else if (key_ && tr_variantIsDict(parent)) { node = tr_variantDictAdd(parent, *key_); key_.reset(); } } return node; } }; bool tr_variantParseBenc(tr_variant& top, int parse_opts, std::string_view benc, char const** setme_end, tr_error** error) { using Stack = transmission::benc::ParserStack<512>; auto stack = Stack{}; auto handler = MyHandler{ &top, parse_opts }; bool const ok = transmission::benc::parse(benc, stack, handler, setme_end, error); if (!ok) { tr_variantFree(&top); } return ok; } /**** ***** ****/ static void saveIntFunc(tr_variant const* val, void* vevbuf) { auto* evbuf = static_cast(vevbuf); evbuffer_add_printf(evbuf, "i%" PRId64 "e", val->val.i); } static void saveBoolFunc(tr_variant const* val, void* vevbuf) { auto* evbuf = static_cast(vevbuf); if (val->val.b) { evbuffer_add(evbuf, "i1e", 3); } else { evbuffer_add(evbuf, "i0e", 3); } } static void saveRealFunc(tr_variant const* val, void* vevbuf) { auto buf = std::array{}; int const len = tr_snprintf(std::data(buf), std::size(buf), "%f", val->val.d); auto* evbuf = static_cast(vevbuf); evbuffer_add_printf(evbuf, "%d:", len); evbuffer_add(evbuf, std::data(buf), len); } static void saveStringFunc(tr_variant const* v, void* vevbuf) { auto sv = std::string_view{}; (void)!tr_variantGetStrView(v, &sv); auto* evbuf = static_cast(vevbuf); evbuffer_add_printf(evbuf, "%zu:", std::size(sv)); evbuffer_add(evbuf, std::data(sv), std::size(sv)); } static void saveDictBeginFunc(tr_variant const* /*val*/, void* vevbuf) { auto* evbuf = static_cast(vevbuf); evbuffer_add(evbuf, "d", 1); } static void saveListBeginFunc(tr_variant const* /*val*/, void* vevbuf) { auto* evbuf = static_cast(vevbuf); evbuffer_add(evbuf, "l", 1); } static void saveContainerEndFunc(tr_variant const* /*val*/, void* vevbuf) { auto* evbuf = static_cast(vevbuf); evbuffer_add(evbuf, "e", 1); } static struct VariantWalkFuncs const walk_funcs = { saveIntFunc, // saveBoolFunc, // saveRealFunc, // saveStringFunc, // saveDictBeginFunc, // saveListBeginFunc, // saveContainerEndFunc, // }; void tr_variantToBufBenc(tr_variant const* top, struct evbuffer* buf) { tr_variantWalk(top, &walk_funcs, buf, true); }