#include #include #include #include #include #include #include namespace scwx { namespace wsr88d { namespace rpg { static const std::string logPrefix_ = "[scwx::wsr88d::rpg::tabular_alphanumeric_block] "; class TabularAlphanumericBlockImpl { public: explicit TabularAlphanumericBlockImpl() : blockDivider1_ {0}, blockId_ {0}, lengthOfBlock_ {0}, messageHeader_ {}, descriptionBlock_ {}, blockDivider2_ {0}, numberOfPages_ {0}, pageList_ {} { } ~TabularAlphanumericBlockImpl() = default; int16_t blockDivider1_; int16_t blockId_; uint32_t lengthOfBlock_; std::shared_ptr messageHeader_; std::shared_ptr descriptionBlock_; int16_t blockDivider2_; uint16_t numberOfPages_; std::vector> pageList_; }; TabularAlphanumericBlock::TabularAlphanumericBlock() : Message(), p(std::make_unique()) { } TabularAlphanumericBlock::~TabularAlphanumericBlock() = default; TabularAlphanumericBlock::TabularAlphanumericBlock( TabularAlphanumericBlock&&) noexcept = default; TabularAlphanumericBlock& TabularAlphanumericBlock::operator=( TabularAlphanumericBlock&&) noexcept = default; int16_t TabularAlphanumericBlock::block_divider() const { return p->blockDivider1_; } size_t TabularAlphanumericBlock::data_size() const { return p->lengthOfBlock_; } bool TabularAlphanumericBlock::Parse(std::istream& is) { bool blockValid = true; const std::streampos blockStart = is.tellg(); is.read(reinterpret_cast(&p->blockDivider1_), 2); is.read(reinterpret_cast(&p->blockId_), 2); is.read(reinterpret_cast(&p->lengthOfBlock_), 4); p->blockDivider1_ = ntohs(p->blockDivider1_); p->blockId_ = ntohs(p->blockId_); p->lengthOfBlock_ = ntohl(p->lengthOfBlock_); if (is.eof()) { BOOST_LOG_TRIVIAL(debug) << logPrefix_ << "Reached end of file"; blockValid = false; } else { if (p->blockDivider1_ != -1) { BOOST_LOG_TRIVIAL(warning) << logPrefix_ << "Invalid first block divider: " << p->blockDivider1_; blockValid = false; } if (p->blockId_ != 3) { BOOST_LOG_TRIVIAL(warning) << logPrefix_ << "Invalid block ID: " << p->blockId_; blockValid = false; } if (p->lengthOfBlock_ < 10) { BOOST_LOG_TRIVIAL(warning) << logPrefix_ << "Invalid block length: " << p->lengthOfBlock_; blockValid = false; } } if (blockValid) { p->messageHeader_ = std::make_shared(); blockValid = p->messageHeader_->Parse(is); if (!blockValid) { p->messageHeader_ = nullptr; } } if (blockValid) { p->descriptionBlock_ = std::make_shared(); blockValid = p->descriptionBlock_->Parse(is); if (!blockValid) { p->descriptionBlock_ = nullptr; } } if (blockValid) { is.read(reinterpret_cast(&p->blockDivider2_), 2); is.read(reinterpret_cast(&p->numberOfPages_), 2); p->blockDivider2_ = ntohs(p->blockDivider2_); p->numberOfPages_ = ntohs(p->numberOfPages_); if (p->blockDivider2_ != -1) { BOOST_LOG_TRIVIAL(warning) << logPrefix_ << "Invalid second block divider: " << p->blockDivider2_; blockValid = false; } if (p->numberOfPages_ < 1 || p->numberOfPages_ > 48) { BOOST_LOG_TRIVIAL(warning) << logPrefix_ << "Invalid number of pages: " << p->numberOfPages_; blockValid = false; } } if (blockValid) { uint16_t numberOfCharacters; for (uint16_t i = 0; i < p->numberOfPages_; i++) { std::vector lineList; while (!is.eof()) { is.read(reinterpret_cast(&numberOfCharacters), 2); numberOfCharacters = ntohs(numberOfCharacters); if (static_cast(numberOfCharacters) == -1) { // End of page break; } else if (numberOfCharacters > 80) { BOOST_LOG_TRIVIAL(warning) << logPrefix_ << "Invalid number of characters: " << numberOfCharacters << " (Page " << (i + 1) << ")"; blockValid = false; break; } std::string line; line.resize(numberOfCharacters); is.read(line.data(), numberOfCharacters); lineList.push_back(std::move(line)); } p->pageList_.push_back(std::move(lineList)); } } const std::streampos blockEnd = is.tellg(); if (!ValidateMessage(is, blockEnd - blockStart)) { blockValid = false; } return blockValid; } } // namespace rpg } // namespace wsr88d } // namespace scwx