Codebase list minetestmapper / upstream/20180325 PlayerAttributes.cpp
upstream/20180325

Tree @upstream/20180325 (Download .tar.gz)

PlayerAttributes.cpp @upstream/20180325raw · history · blame

#include <fstream>
#include <sstream>
#include <stdexcept>
#include <dirent.h>
#include <unistd.h> // for usleep
#include <sqlite3.h>

#include "config.h"
#include "PlayerAttributes.h"
#include "util.h"

using namespace std;

PlayerAttributes::PlayerAttributes(const std::string &worldDir)
{
	std::ifstream ifs((worldDir + "world.mt").c_str());
	if (!ifs.good())
		throw std::runtime_error("Failed to read world.mt");
	std::string backend = read_setting_default("player_backend", ifs, "files");
	ifs.close();

	if (backend == "files")
		readFiles(worldDir + "players");
	else if (backend == "sqlite3")
		readSqlite(worldDir + "players.sqlite");
	else
		throw std::runtime_error(((std::string) "Unknown player backend: ") + backend);
}

void PlayerAttributes::readFiles(const std::string &playersPath)
{
	DIR *dir;
	dir = opendir (playersPath.c_str());
	if (dir == NULL)
		return;

	struct dirent *ent;
	while ((ent = readdir (dir)) != NULL) {
		if (ent->d_name[0] == '.')
			continue;

		string path = playersPath + PATH_SEPARATOR + ent->d_name;
		ifstream in(path.c_str());
		if(!in.good())
			continue;

		string name, position;
		name = read_setting("name", in);
		in.seekg(0);
		position = read_setting("position", in);

		Player player;
		istringstream iss(position);
		char tmp;
		iss >> tmp; // '('
		iss >> player.x;
		iss >> tmp; // ','
		iss >> player.y;
		iss >> tmp; // ','
		iss >> player.z;
		iss >> tmp; // ')'
		if(tmp != ')')
			continue;
		player.name = name;

		player.x /= 10.0;
		player.y /= 10.0;
		player.z /= 10.0;

		m_players.push_back(player);
	}
	closedir(dir);
}

/**********/

#define SQLRES(f, good) \
	result = (sqlite3_##f); \
	if (result != good) { \
		throw std::runtime_error(sqlite3_errmsg(db));\
	}
#define SQLOK(f) SQLRES(f, SQLITE_OK)

void PlayerAttributes::readSqlite(const std::string &db_name)
{
	int result;
	sqlite3 *db;
	sqlite3_stmt *stmt_get_player_pos;

	SQLOK(open_v2(db_name.c_str(), &db, SQLITE_OPEN_READONLY |
			SQLITE_OPEN_PRIVATECACHE, 0))

	SQLOK(prepare_v2(db,
			"SELECT name, posX, posY, posZ FROM player",
		-1, &stmt_get_player_pos, NULL))

	while ((result = sqlite3_step(stmt_get_player_pos)) != SQLITE_DONE) {
		if (result == SQLITE_BUSY) { // Wait some time and try again
			usleep(10000);
		} else if (result != SQLITE_ROW) {
			throw std::runtime_error(sqlite3_errmsg(db));
		}

		Player player;
		const unsigned char *name_ = sqlite3_column_text(stmt_get_player_pos, 0);
		player.name = std::string(reinterpret_cast<const char*>(name_));
		player.x = sqlite3_column_double(stmt_get_player_pos, 1);
		player.y = sqlite3_column_double(stmt_get_player_pos, 2);
		player.z = sqlite3_column_double(stmt_get_player_pos, 3);

		player.x /= 10.0;
		player.y /= 10.0;
		player.z /= 10.0;

		m_players.push_back(player);
	}

	sqlite3_finalize(stmt_get_player_pos);
	sqlite3_close(db);
}

/**********/

PlayerAttributes::Players::iterator PlayerAttributes::begin()
{
	return m_players.begin();
}

PlayerAttributes::Players::iterator PlayerAttributes::end()
{
	return m_players.end();
}