/* * This file is part of the Flowee project * Copyright (C) 2020-2021 Tom Zander * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #include "P2PParser.h" #include Streaming::ParsingException::ParsingException(const char *message) : std::runtime_error(message) { } Streaming::P2PParser::P2PParser(const Message &message) : m_constBuffer(message.body()), m_privData(m_constBuffer.begin()), m_data(m_privData), m_end(m_constBuffer.end()) { } Streaming::P2PParser::P2PParser(const Streaming::ConstBuffer &data) : m_constBuffer(data), m_privData(m_constBuffer.begin()), m_data(m_privData), m_end(m_constBuffer.end()) { } std::string Streaming::P2PParser::readString() { // first read a varint for size then return the string of that size if (m_data > m_end) throw ParsingException("Out of range"); size_t size = readCompactInt(); if (size > remaining()) throw ParsingException("Out of range"); auto ptr = m_data; m_data += size; return std::string(ptr, size); } uint8_t Streaming::P2PParser::readByte() { if (m_data + 1 > m_end) throw ParsingException("Out of range"); return static_cast(*m_data++); } uint16_t Streaming::P2PParser::readWord() { if (m_data + 2 > m_end) throw ParsingException("Out of range"); const unsigned char *ptr = reinterpret_cast(m_data); m_data += 2; return ReadLE16(ptr); } uint16_t Streaming::P2PParser::readWordBE() { if (m_data + 2 > m_end) throw ParsingException("Out of range"); const unsigned char *ptr = reinterpret_cast(m_data); m_data += 2; return be16toh(*(uint16_t*) ptr); } uint32_t Streaming::P2PParser::readInt() { if (m_data + 4 > m_end) throw ParsingException("Out of range"); const unsigned char *ptr = reinterpret_cast(m_data); m_data += 4; return ReadLE32(ptr); } uint64_t Streaming::P2PParser::readLong() { if (m_data + 8 > m_end) throw ParsingException("Out of range"); const unsigned char *ptr = reinterpret_cast(m_data); m_data += 8; return ReadLE64(ptr); } uint64_t Streaming::P2PParser::readCompactInt() { if (m_data >= m_end) throw ParsingException("Out of range"); uint64_t answer = static_cast(*m_data++); switch (answer) { case 253: return readWord(); case 254: return readInt(); case 255: return readLong(); default: return answer; } } bool Streaming::P2PParser::readBool() { if (m_data + 1 > m_end) throw ParsingException("Out of range"); return *m_data++ == true; } std::vector Streaming::P2PParser::readBytes(int32_t count) { assert(count > 0); if (count < 0 || static_cast(count) > remaining()) throw ParsingException("Out of range"); std::vector answer; answer.resize(count); for (int i = 0; i < count; ++i) { answer[i] = m_data[i]; } m_data += count; return answer; } std::vector Streaming::P2PParser::readUnsignedBytes(int32_t count) { assert(count > 0); if (count < 0 || static_cast(count) > remaining()) throw ParsingException("Out of range"); std::vector answer; answer.resize(count); for (int i = 0; i < count; ++i) { answer[i] = m_data[i]; } m_data += count; return answer; } uint256 Streaming::P2PParser::readUint256() { if (m_data + 32 > m_end) throw ParsingException("Out of range"); uint256 answer(m_data); m_data += 32; return answer; }