#include "WorldSave.hpp"
#include "filesystem.hpp"
+#include "TokenStreamReader.hpp"
#include <cctype>
#include <cstring>
#include <limits>
#include <stdexcept>
#include <zlib.h>
+#include <glm/gtx/io.hpp>
using namespace std;
: root_path(path)
, world_conf_path(path + "world.conf")
, gen_conf_path(path + "gen.conf")
+, player_path(path + "player/")
, chunk_path(path + "chunks/%d/%d/%d.gz")
, chunk_bufsiz(chunk_path.length() + 3 * std::numeric_limits<int>::digits10)
, chunk_buf(new char[chunk_bufsiz]) {
}
-// TODO: better implementation of config files
void WorldSave::Read(World::Config &conf) const {
- ifstream in(world_conf_path);
- if (!in) {
+ ifstream is(world_conf_path);
+ if (!is) {
throw runtime_error("failed to open world config");
}
+ TokenStreamReader in(is);
- constexpr char spaces[] = "\n\r\t ";
-
- string line;
- while (getline(in, line)) {
- if (line.empty() || line[0] == '#') continue;
- auto equals_pos = line.find_first_of('=');
-
- auto name_begin = line.find_first_not_of(spaces, 0, sizeof(spaces));
- auto name_end = equals_pos - 1;
- while (name_end > name_begin && isspace(line[name_end])) {
- --name_end;
+ string name;
+ while (in.HasMore()) {
+ in.ReadIdentifier(name);
+ in.Skip(Token::EQUALS);
+ if (name == "spawn") {
+ in.ReadVec(conf.spawn);
}
-
- auto value_begin = line.find_first_not_of(spaces, equals_pos + 1, sizeof(spaces));
- auto value_end = line.length() - 1;
- while (value_end > value_begin && isspace(line[value_end])) {
- --value_end;
+ if (in.HasMore() && in.Peek().type == Token::SEMICOLON) {
+ in.Skip(Token::SEMICOLON);
}
-
- string name(line, name_begin, name_end - name_begin + 1);
- string value(line, value_begin, value_end - value_begin + 1);
-
- // if (name == "seed") {
- // conf.gen.seed = stoul(value);
- // } else {
- throw runtime_error("unknown world option: " + name);
- // }
}
- if (in.bad()) {
+
+ if (is.bad()) {
throw runtime_error("IO error reading world config");
}
}
}
ofstream out(world_conf_path);
- //out << "seed = " << conf.gen.seed << endl;
+ out << "spawn = " << conf.spawn << ';' << endl;
out.close();
if (!out) {
void WorldSave::Read(Generator::Config &conf) const {
- ifstream in(gen_conf_path);
- if (!in) {
+ ifstream is(gen_conf_path);
+ if (!is) {
throw runtime_error("failed to open generator config");
}
+ TokenStreamReader in(is);
- constexpr char spaces[] = "\n\r\t ";
-
- string line;
- while (getline(in, line)) {
- if (line.empty() || line[0] == '#') continue;
- auto equals_pos = line.find_first_of('=');
-
- auto name_begin = line.find_first_not_of(spaces, 0, sizeof(spaces));
- auto name_end = equals_pos - 1;
- while (name_end > name_begin && isspace(line[name_end])) {
- --name_end;
- }
-
- auto value_begin = line.find_first_not_of(spaces, equals_pos + 1, sizeof(spaces));
- auto value_end = line.length() - 1;
- while (value_end > value_begin && isspace(line[value_end])) {
- --value_end;
- }
-
- string name(line, name_begin, name_end - name_begin + 1);
- string value(line, value_begin, value_end - value_begin + 1);
-
+ string name;
+ while (in.HasMore()) {
+ in.ReadIdentifier(name);
+ in.Skip(Token::EQUALS);
if (name == "seed") {
- conf.seed = stoul(value);
- } else {
- throw runtime_error("unknown generator option: " + name);
+ in.ReadNumber(conf.seed);
+ }
+ if (in.HasMore() && in.Peek().type == Token::SEMICOLON) {
+ in.Skip(Token::SEMICOLON);
}
}
- if (in.bad()) {
- throw runtime_error("IO error reading world config");
+
+ if (is.bad()) {
+ throw runtime_error("IO error reading generator config");
}
}
}
ofstream out(gen_conf_path);
- out << "seed = " << conf.seed << endl;
+ out << "seed = " << conf.seed << ';' << endl;
out.close();
if (!out) {
}
-bool WorldSave::Exists(const Chunk::Pos &pos) const noexcept {
+bool WorldSave::Exists(const Player &player) const {
+ return is_file(PlayerPath(player));
+}
+
+void WorldSave::Read(Player &player) const {
+ ifstream is(PlayerPath(player));
+ TokenStreamReader in(is);
+ string name;
+ EntityState state;
+ while (in.HasMore()) {
+ in.ReadIdentifier(name);
+ in.Skip(Token::EQUALS);
+ if (name == "chunk") {
+ in.ReadVec(state.pos.chunk);
+ } else if (name == "position") {
+ in.ReadVec(state.pos.block);
+ } else if (name == "orientation") {
+ in.ReadQuat(state.orient);
+ } else if (name == "pitch") {
+ state.pitch = in.GetFloat();
+ } else if (name == "yaw") {
+ state.yaw = in.GetFloat();
+ } else if (name == "slot") {
+ int slot;
+ in.ReadNumber(slot);
+ player.SetInventorySlot(slot);
+ }
+ if (in.HasMore() && in.Peek().type == Token::SEMICOLON) {
+ in.Skip(Token::SEMICOLON);
+ }
+ }
+ player.GetEntity().SetState(state);
+}
+
+void WorldSave::Write(const Player &player) const {
+ if (!make_dirs(player_path)) {
+ throw runtime_error("failed to create player save directory");
+ }
+ const EntityState &state = player.GetEntity().GetState();
+ ofstream out(PlayerPath(player));
+ out << "chunk = " << state.pos.chunk << ';' << endl;
+ out << "position = " << state.pos.block << ';' << endl;
+ out << "orientation = " << state.orient << ';' << endl;
+ out << "pitch = " << state.pitch << ';' << endl;
+ out << "yaw = " << state.yaw << ';' << endl;
+ out << "slot = " << player.GetInventorySlot() << ';' << endl;
+}
+
+string WorldSave::PlayerPath(const Player &player) const {
+ // TODO: this is potentially dangerous, server and client should
+ // provide a sanitized name for storage
+ return player_path + player.Name();
+}
+
+
+bool WorldSave::Exists(const ExactLocation::Coarse &pos) const noexcept {
return is_file(ChunkPath(pos));
}
if (gzclose(file) != Z_OK) {
throw runtime_error("failed to read chunk file");
}
- chunk.InvalidateModel();
+ chunk.InvalidateMesh();
chunk.ClearSave();
}
}
-const char *WorldSave::ChunkPath(const Chunk::Pos &pos) const {
+const char *WorldSave::ChunkPath(const ExactLocation::Coarse &pos) const {
snprintf(chunk_buf.get(), chunk_bufsiz, chunk_path.c_str(), pos.x, pos.y, pos.z);
return chunk_buf.get();
}