extend user block fields check
This commit is contained in:
parent
0ad588fd33
commit
ab53922474
@ -24,6 +24,7 @@
|
|||||||
#include "data/StructLayout.hpp"
|
#include "data/StructLayout.hpp"
|
||||||
|
|
||||||
namespace fs = std::filesystem;
|
namespace fs = std::filesystem;
|
||||||
|
using namespace data;
|
||||||
|
|
||||||
static debug::Logger logger("content-loader");
|
static debug::Logger logger("content-loader");
|
||||||
|
|
||||||
@ -117,6 +118,30 @@ void ContentLoader::fixPackIndices() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void perform_user_block_fields(
|
||||||
|
const std::string& blockName, StructLayout& layout
|
||||||
|
) {
|
||||||
|
if (layout.size() > MAX_USER_BLOCK_FIELDS_SIZE) {
|
||||||
|
throw std::runtime_error(
|
||||||
|
util::quote(blockName) +
|
||||||
|
" fields total size exceeds limit (" +
|
||||||
|
std::to_string(layout.size()) + "/" +
|
||||||
|
std::to_string(MAX_USER_BLOCK_FIELDS_SIZE) + ")");
|
||||||
|
}
|
||||||
|
for (const auto& field : layout) {
|
||||||
|
if (field.name.at(0) == '.') {
|
||||||
|
throw std::runtime_error(
|
||||||
|
util::quote(blockName) + " field " + field.name +
|
||||||
|
": user field may not start with '.'");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
std::vector<Field> fields;
|
||||||
|
fields.insert(fields.end(), layout.begin(), layout.end());
|
||||||
|
// add built-in fields here
|
||||||
|
layout = StructLayout::create(fields);
|
||||||
|
}
|
||||||
|
|
||||||
void ContentLoader::loadBlock(
|
void ContentLoader::loadBlock(
|
||||||
Block& def, const std::string& name, const fs::path& file
|
Block& def, const std::string& name, const fs::path& file
|
||||||
) {
|
) {
|
||||||
@ -253,15 +278,10 @@ void ContentLoader::loadBlock(
|
|||||||
root.at("tick-interval").get(def.tickInterval);
|
root.at("tick-interval").get(def.tickInterval);
|
||||||
|
|
||||||
if (root.has("fields")) {
|
if (root.has("fields")) {
|
||||||
def.dataStruct = std::make_unique<data::StructLayout>();
|
def.dataStruct = std::make_unique<StructLayout>();
|
||||||
def.dataStruct->deserialize(root["fields"]);
|
def.dataStruct->deserialize(root["fields"]);
|
||||||
if (def.dataStruct->size() > MAX_BLOCK_FIELDS_SIZE) {
|
|
||||||
throw std::runtime_error(
|
perform_user_block_fields(def.name, *def.dataStruct);
|
||||||
util::quote(def.name) +
|
|
||||||
" fields total size exceeds limit (" +
|
|
||||||
std::to_string(def.dataStruct->size()) + "/" +
|
|
||||||
std::to_string(MAX_BLOCK_FIELDS_SIZE) + ")");
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (def.tickInterval == 0) {
|
if (def.tickInterval == 0) {
|
||||||
|
|||||||
@ -385,7 +385,7 @@ void StructLayout::deserialize(const dv::value& src) {
|
|||||||
fieldmap["convert-strategy"].asString()
|
fieldmap["convert-strategy"].asString()
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
fields.push_back(Field {type, name, elements, convertStrategy});
|
fields.push_back(Field (type, name, elements, convertStrategy));
|
||||||
}
|
}
|
||||||
*this = create(fields);
|
*this = create(fields);
|
||||||
}
|
}
|
||||||
|
|||||||
@ -81,6 +81,18 @@ namespace data {
|
|||||||
/// @brief Byte size of the field
|
/// @brief Byte size of the field
|
||||||
int size;
|
int size;
|
||||||
|
|
||||||
|
Field(
|
||||||
|
FieldType type,
|
||||||
|
std::string name,
|
||||||
|
int elements,
|
||||||
|
FieldConvertStrategy strategy=FieldConvertStrategy::RESET
|
||||||
|
) : type(type),
|
||||||
|
name(std::move(name)),
|
||||||
|
elements(elements),
|
||||||
|
convertStrategy(strategy),
|
||||||
|
offset(0),
|
||||||
|
size(0) {}
|
||||||
|
|
||||||
bool operator==(const Field& o) const {
|
bool operator==(const Field& o) const {
|
||||||
return type == o.type &&
|
return type == o.type &&
|
||||||
name == o.name &&
|
name == o.name &&
|
||||||
@ -233,6 +245,14 @@ namespace data {
|
|||||||
return totalSize;
|
return totalSize;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
[[nodiscard]] const auto begin() const {
|
||||||
|
return fields.begin();
|
||||||
|
}
|
||||||
|
|
||||||
|
[[nodiscard]] const auto end() const {
|
||||||
|
return fields.end();
|
||||||
|
}
|
||||||
|
|
||||||
/// @brief Convert structure data from srcLayout to this layout.
|
/// @brief Convert structure data from srcLayout to this layout.
|
||||||
/// @param srcLayout source structure layout
|
/// @param srcLayout source structure layout
|
||||||
/// @param src source data
|
/// @param src source data
|
||||||
|
|||||||
@ -1,5 +1,6 @@
|
|||||||
#include "Block.hpp"
|
#include "Block.hpp"
|
||||||
|
|
||||||
|
#include <set>
|
||||||
#include <utility>
|
#include <utility>
|
||||||
|
|
||||||
#include "core_defs.hpp"
|
#include "core_defs.hpp"
|
||||||
@ -141,3 +142,10 @@ void Block::cloneTo(Block& dst) {
|
|||||||
dst.inventorySize = inventorySize;
|
dst.inventorySize = inventorySize;
|
||||||
dst.tickInterval = tickInterval;
|
dst.tickInterval = tickInterval;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static std::set<std::string, std::less<>> RESERVED_BLOCK_FIELDS {
|
||||||
|
};
|
||||||
|
|
||||||
|
bool Block::isReservedBlockField(std::string_view view) {
|
||||||
|
return RESERVED_BLOCK_FIELDS.find(view) != RESERVED_BLOCK_FIELDS.end();
|
||||||
|
}
|
||||||
|
|||||||
@ -26,7 +26,7 @@ inline constexpr uint FACE_PZ = 5;
|
|||||||
/// complex hitboxes
|
/// complex hitboxes
|
||||||
inline constexpr uint BLOCK_AABB_GRID = 16;
|
inline constexpr uint BLOCK_AABB_GRID = 16;
|
||||||
|
|
||||||
inline constexpr size_t MAX_BLOCK_FIELDS_SIZE = 240;
|
inline constexpr size_t MAX_USER_BLOCK_FIELDS_SIZE = 240;
|
||||||
|
|
||||||
inline std::string DEFAULT_MATERIAL = "base:stone";
|
inline std::string DEFAULT_MATERIAL = "base:stone";
|
||||||
|
|
||||||
@ -222,6 +222,8 @@ public:
|
|||||||
~Block();
|
~Block();
|
||||||
|
|
||||||
void cloneTo(Block& dst);
|
void cloneTo(Block& dst);
|
||||||
|
|
||||||
|
static bool isReservedBlockField(std::string_view view);
|
||||||
};
|
};
|
||||||
|
|
||||||
inline glm::ivec3 get_ground_direction(const Block& def, int rotation) {
|
inline glm::ivec3 get_ground_direction(const Block& def, int rotation) {
|
||||||
|
|||||||
@ -117,9 +117,9 @@ TEST(StructLayout, ConvertWithLoss) {
|
|||||||
|
|
||||||
TEST(StructLayout, Serialization) {
|
TEST(StructLayout, Serialization) {
|
||||||
std::vector<Field> fields {
|
std::vector<Field> fields {
|
||||||
Field {FieldType::CHAR, "text", 5},
|
Field (FieldType::CHAR, "text", 5),
|
||||||
Field {FieldType::I16, "someint", 1},
|
Field (FieldType::I16, "someint", 1),
|
||||||
Field {FieldType::F64, "pi", 1},
|
Field (FieldType::F64, "pi", 1),
|
||||||
};
|
};
|
||||||
auto layout1 = StructLayout::create(fields);
|
auto layout1 = StructLayout::create(fields);
|
||||||
auto serialized = layout1.serialize();
|
auto serialized = layout1.serialize();
|
||||||
|
|||||||
Loading…
x
Reference in New Issue
Block a user