2019-05-02 20:24:11 +08:00
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include <cstdint>
|
|
|
|
#include <iostream>
|
|
|
|
#include <memory>
|
|
|
|
#include <vector>
|
|
|
|
|
|
|
|
#include <fmt/format.h>
|
|
|
|
#include <gtest/gtest.h>
|
|
|
|
|
|
|
|
#include "slk/streams.hpp"
|
2021-01-21 22:47:56 +08:00
|
|
|
#include "utils/logging.hpp"
|
2019-05-02 20:24:11 +08:00
|
|
|
|
|
|
|
namespace slk {
|
|
|
|
|
|
|
|
/// Class used for SLK tests. It creates a `slk::Builder` that can be written
|
|
|
|
/// to. After you have written the data to the builder, you can get a
|
|
|
|
/// `slk::Reader` and try to decode the encoded data.
|
|
|
|
class Loopback {
|
|
|
|
public:
|
|
|
|
~Loopback() {
|
2021-01-21 22:47:56 +08:00
|
|
|
MG_ASSERT(builder_, "You haven't created a builder!");
|
|
|
|
MG_ASSERT(reader_, "You haven't created a reader!");
|
2019-05-02 20:24:11 +08:00
|
|
|
reader_->Finalize();
|
|
|
|
}
|
|
|
|
|
|
|
|
slk::Builder *GetBuilder() {
|
2021-01-21 22:47:56 +08:00
|
|
|
MG_ASSERT(!builder_, "You have already allocated a builder!");
|
2019-05-02 20:24:11 +08:00
|
|
|
builder_ = std::make_unique<slk::Builder>(
|
2021-02-18 22:32:43 +08:00
|
|
|
[this](const uint8_t *data, size_t size, bool have_more) { Write(data, size, have_more); });
|
2019-05-02 20:24:11 +08:00
|
|
|
return builder_.get();
|
|
|
|
}
|
|
|
|
|
|
|
|
slk::Reader *GetReader() {
|
2021-02-18 22:32:43 +08:00
|
|
|
MG_ASSERT(builder_, "You must first get a builder before getting a reader!");
|
2021-01-21 22:47:56 +08:00
|
|
|
MG_ASSERT(!reader_, "You have already allocated a reader!");
|
2019-05-02 20:24:11 +08:00
|
|
|
builder_->Finalize();
|
|
|
|
auto ret = slk::CheckStreamComplete(data_.data(), data_.size());
|
2021-01-21 22:47:56 +08:00
|
|
|
MG_ASSERT(ret.status == slk::StreamStatus::COMPLETE);
|
|
|
|
MG_ASSERT(ret.stream_size == data_.size());
|
2019-05-02 20:24:11 +08:00
|
|
|
size_ = ret.encoded_data_size;
|
|
|
|
Dump();
|
|
|
|
reader_ = std::make_unique<slk::Reader>(data_.data(), data_.size());
|
|
|
|
return reader_.get();
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t size() { return size_; }
|
|
|
|
|
|
|
|
private:
|
|
|
|
void Write(const uint8_t *data, size_t size, bool have_more) {
|
|
|
|
for (size_t i = 0; i < size; ++i) {
|
|
|
|
data_.push_back(data[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Dump() {
|
|
|
|
std::string dump;
|
|
|
|
for (size_t i = 0; i < data_.size(); ++i) {
|
|
|
|
dump += fmt::format("{:02x}", data_[i]);
|
|
|
|
if (i != data_.size() - 1) {
|
|
|
|
dump += " ";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// This stores the encoded SLK stream into the test XML output. To get the
|
|
|
|
// data you have to specify to the test (during runtime) that it should
|
|
|
|
// create an XML output.
|
|
|
|
::testing::Test::RecordProperty("slk_stream", dump);
|
|
|
|
}
|
|
|
|
|
|
|
|
std::vector<uint8_t> data_;
|
|
|
|
std::unique_ptr<slk::Builder> builder_;
|
|
|
|
std::unique_ptr<slk::Reader> reader_;
|
|
|
|
size_t size_{0};
|
|
|
|
};
|
|
|
|
|
|
|
|
} // namespace slk
|