2010-11-29 19:13:04 +01:00
|
|
|
/*
|
2013-02-24 18:40:43 +01:00
|
|
|
Minetest
|
2013-02-24 19:38:45 +01:00
|
|
|
Copyright (C) 2010-2013 celeron55, Perttu Ahola <celeron55@gmail.com>
|
2010-11-29 19:13:04 +01:00
|
|
|
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
2012-06-05 16:56:56 +02:00
|
|
|
it under the terms of the GNU Lesser General Public License as published by
|
|
|
|
the Free Software Foundation; either version 2.1 of the License, or
|
2010-11-29 19:13:04 +01:00
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
2012-06-05 16:56:56 +02:00
|
|
|
GNU Lesser General Public License for more details.
|
2010-11-29 19:13:04 +01:00
|
|
|
|
2012-06-05 16:56:56 +02:00
|
|
|
You should have received a copy of the GNU Lesser General Public License along
|
2010-11-29 19:13:04 +01:00
|
|
|
with this program; if not, write to the Free Software Foundation, Inc.,
|
|
|
|
51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
|
|
*/
|
|
|
|
|
2010-11-27 00:02:21 +01:00
|
|
|
#include "server.h"
|
|
|
|
#include <iostream>
|
2011-11-27 10:44:05 +01:00
|
|
|
#include <queue>
|
2013-01-23 18:32:02 +01:00
|
|
|
#include <algorithm>
|
2017-08-24 08:28:54 +02:00
|
|
|
#include "network/connection.h"
|
2015-01-12 17:01:41 +01:00
|
|
|
#include "network/networkprotocol.h"
|
|
|
|
#include "network/serveropcodes.h"
|
2013-08-11 04:09:45 +02:00
|
|
|
#include "ban.h"
|
|
|
|
#include "environment.h"
|
2010-11-27 00:02:21 +01:00
|
|
|
#include "map.h"
|
2015-04-07 12:13:12 +02:00
|
|
|
#include "threading/mutex_auto_lock.h"
|
2010-11-27 00:02:21 +01:00
|
|
|
#include "constants.h"
|
2010-12-11 17:11:03 +01:00
|
|
|
#include "voxel.h"
|
2011-04-24 23:31:22 +02:00
|
|
|
#include "config.h"
|
2013-09-25 04:29:07 +02:00
|
|
|
#include "version.h"
|
2011-05-22 16:00:09 +02:00
|
|
|
#include "filesys.h"
|
2011-06-26 01:34:36 +02:00
|
|
|
#include "mapblock.h"
|
2011-06-26 14:48:56 +02:00
|
|
|
#include "serverobject.h"
|
2013-08-13 23:06:39 +02:00
|
|
|
#include "genericobject.h"
|
2011-10-12 12:53:38 +02:00
|
|
|
#include "settings.h"
|
|
|
|
#include "profiler.h"
|
2011-10-16 13:57:53 +02:00
|
|
|
#include "log.h"
|
2017-04-25 19:38:08 +02:00
|
|
|
#include "scripting_server.h"
|
2011-11-14 20:41:30 +01:00
|
|
|
#include "nodedef.h"
|
2012-01-12 06:10:39 +01:00
|
|
|
#include "itemdef.h"
|
2011-11-17 01:28:46 +01:00
|
|
|
#include "craftdef.h"
|
2013-02-14 04:43:15 +01:00
|
|
|
#include "emerge.h"
|
2017-11-08 23:56:20 +01:00
|
|
|
#include "mapgen/mapgen.h"
|
|
|
|
#include "mapgen/mg_biome.h"
|
2017-06-06 16:19:04 +02:00
|
|
|
#include "content_mapnode.h"
|
2012-03-19 03:04:16 +01:00
|
|
|
#include "content_nodemeta.h"
|
|
|
|
#include "content_sao.h"
|
2018-04-17 15:54:50 +02:00
|
|
|
#include "content/mods.h"
|
2017-09-26 00:11:20 +02:00
|
|
|
#include "modchannels.h"
|
2013-02-21 23:00:44 +01:00
|
|
|
#include "serverlist.h"
|
2012-06-17 01:40:36 +02:00
|
|
|
#include "util/string.h"
|
2012-07-26 21:06:45 +02:00
|
|
|
#include "rollback.h"
|
2012-11-26 10:18:34 +01:00
|
|
|
#include "util/serialize.h"
|
2013-08-11 04:09:45 +02:00
|
|
|
#include "util/thread.h"
|
2013-03-21 20:42:23 +01:00
|
|
|
#include "defaultsettings.h"
|
2018-03-16 08:41:33 +01:00
|
|
|
#include "server/mods.h"
|
2015-02-21 14:17:36 +01:00
|
|
|
#include "util/base64.h"
|
|
|
|
#include "util/sha1.h"
|
|
|
|
#include "util/hex.h"
|
2017-11-08 23:56:20 +01:00
|
|
|
#include "database/database.h"
|
2017-07-16 10:47:31 +02:00
|
|
|
#include "chatmessage.h"
|
2017-08-16 22:11:45 +02:00
|
|
|
#include "chat_interface.h"
|
2017-08-18 19:25:07 +02:00
|
|
|
#include "remoteplayer.h"
|
2011-10-16 13:57:53 +02:00
|
|
|
|
2013-08-04 07:17:07 +02:00
|
|
|
class ClientNotFoundException : public BaseException
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
ClientNotFoundException(const char *s):
|
|
|
|
BaseException(s)
|
|
|
|
{}
|
|
|
|
};
|
|
|
|
|
2015-04-07 12:13:12 +02:00
|
|
|
class ServerThread : public Thread
|
2013-08-11 04:09:45 +02:00
|
|
|
{
|
|
|
|
public:
|
|
|
|
|
|
|
|
ServerThread(Server *server):
|
2015-04-07 12:13:12 +02:00
|
|
|
Thread("Server"),
|
2013-08-11 04:09:45 +02:00
|
|
|
m_server(server)
|
2015-04-07 12:13:12 +02:00
|
|
|
{}
|
2013-08-11 04:09:45 +02:00
|
|
|
|
2015-04-07 12:13:12 +02:00
|
|
|
void *run();
|
|
|
|
|
|
|
|
private:
|
|
|
|
Server *m_server;
|
2013-08-11 04:09:45 +02:00
|
|
|
};
|
|
|
|
|
2015-04-07 12:13:12 +02:00
|
|
|
void *ServerThread::run()
|
2010-11-27 00:02:21 +01:00
|
|
|
{
|
2010-12-27 13:34:17 +01:00
|
|
|
BEGIN_DEBUG_EXCEPTION_HANDLER
|
|
|
|
|
2014-01-06 20:05:28 +01:00
|
|
|
m_server->AsyncRunStep(true);
|
|
|
|
|
2015-04-07 12:13:12 +02:00
|
|
|
while (!stopRequested()) {
|
2015-08-01 22:52:28 +02:00
|
|
|
try {
|
2014-01-31 00:24:00 +01:00
|
|
|
m_server->AsyncRunStep();
|
2012-11-26 03:16:48 +01:00
|
|
|
|
2010-11-27 00:02:21 +01:00
|
|
|
m_server->Receive();
|
2014-01-31 00:24:00 +01:00
|
|
|
|
2015-08-01 22:52:28 +02:00
|
|
|
} catch (con::NoIncomingDataException &e) {
|
|
|
|
} catch (con::PeerNotFoundException &e) {
|
2011-10-16 13:57:53 +02:00
|
|
|
infostream<<"Server: PeerNotFoundException"<<std::endl;
|
2015-08-01 22:52:28 +02:00
|
|
|
} catch (ClientNotFoundException &e) {
|
|
|
|
} catch (con::ConnectionBindFailed &e) {
|
2012-03-30 17:42:18 +02:00
|
|
|
m_server->setAsyncFatalError(e.what());
|
2015-08-01 22:52:28 +02:00
|
|
|
} catch (LuaError &e) {
|
2017-01-15 13:36:53 +01:00
|
|
|
m_server->setAsyncFatalError(
|
|
|
|
"ServerThread::run Lua: " + std::string(e.what()));
|
2012-03-30 17:42:18 +02:00
|
|
|
}
|
2010-11-27 00:02:21 +01:00
|
|
|
}
|
2012-11-26 03:16:48 +01:00
|
|
|
|
2015-10-14 08:33:04 +02:00
|
|
|
END_DEBUG_EXCEPTION_HANDLER
|
2010-11-27 00:02:21 +01:00
|
|
|
|
2017-08-15 09:30:31 +02:00
|
|
|
return nullptr;
|
2010-11-27 00:02:21 +01:00
|
|
|
}
|
|
|
|
|
2012-03-19 03:04:16 +01:00
|
|
|
v3f ServerSoundParams::getPos(ServerEnvironment *env, bool *pos_exists) const
|
|
|
|
{
|
|
|
|
if(pos_exists) *pos_exists = false;
|
|
|
|
switch(type){
|
|
|
|
case SSP_LOCAL:
|
|
|
|
return v3f(0,0,0);
|
|
|
|
case SSP_POSITIONAL:
|
|
|
|
if(pos_exists) *pos_exists = true;
|
|
|
|
return pos;
|
|
|
|
case SSP_OBJECT: {
|
|
|
|
if(object == 0)
|
|
|
|
return v3f(0,0,0);
|
|
|
|
ServerActiveObject *sao = env->getActiveObject(object);
|
|
|
|
if(!sao)
|
|
|
|
return v3f(0,0,0);
|
|
|
|
if(pos_exists) *pos_exists = true;
|
|
|
|
return sao->getBasePosition(); }
|
|
|
|
}
|
|
|
|
return v3f(0,0,0);
|
|
|
|
}
|
|
|
|
|
2018-06-13 21:58:34 +02:00
|
|
|
void Server::ShutdownState::reset()
|
|
|
|
{
|
|
|
|
m_timer = 0.0f;
|
|
|
|
message.clear();
|
|
|
|
should_reconnect = false;
|
|
|
|
is_requested = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Server::ShutdownState::trigger(float delay, const std::string &msg, bool reconnect)
|
|
|
|
{
|
|
|
|
m_timer = delay;
|
|
|
|
message = msg;
|
|
|
|
should_reconnect = reconnect;
|
|
|
|
}
|
2012-11-26 03:16:48 +01:00
|
|
|
|
2018-06-13 21:58:34 +02:00
|
|
|
void Server::ShutdownState::tick(float dtime, Server *server)
|
|
|
|
{
|
|
|
|
if (m_timer <= 0.0f)
|
|
|
|
return;
|
|
|
|
|
|
|
|
// Timed shutdown
|
|
|
|
static const float shutdown_msg_times[] =
|
|
|
|
{
|
|
|
|
1, 2, 3, 4, 5, 10, 20, 40, 60, 120, 180, 300, 600, 1200, 1800, 3600
|
|
|
|
};
|
|
|
|
|
|
|
|
// Automated messages
|
|
|
|
if (m_timer < shutdown_msg_times[ARRLEN(shutdown_msg_times) - 1]) {
|
|
|
|
for (float t : shutdown_msg_times) {
|
|
|
|
// If shutdown timer matches an automessage, shot it
|
|
|
|
if (m_timer > t && m_timer - dtime < t) {
|
|
|
|
std::wstring periodicMsg = getShutdownTimerMessage();
|
|
|
|
|
|
|
|
infostream << wide_to_utf8(periodicMsg).c_str() << std::endl;
|
|
|
|
server->SendChatMessage(PEER_ID_INEXISTENT, periodicMsg);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
m_timer -= dtime;
|
|
|
|
if (m_timer < 0.0f) {
|
|
|
|
m_timer = 0.0f;
|
|
|
|
is_requested = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
std::wstring Server::ShutdownState::getShutdownTimerMessage() const
|
|
|
|
{
|
|
|
|
std::wstringstream ws;
|
|
|
|
ws << L"*** Server shutting down in "
|
|
|
|
<< duration_to_string(myround(m_timer)).c_str() << ".";
|
|
|
|
return ws.str();
|
|
|
|
}
|
2010-11-27 00:02:21 +01:00
|
|
|
|
|
|
|
/*
|
|
|
|
Server
|
|
|
|
*/
|
|
|
|
|
|
|
|
Server::Server(
|
2012-03-11 13:54:23 +01:00
|
|
|
const std::string &path_world,
|
2012-03-15 14:20:20 +01:00
|
|
|
const SubgameSpec &gamespec,
|
2014-03-07 01:00:03 +01:00
|
|
|
bool simple_singleplayer_mode,
|
2017-09-28 13:47:30 +02:00
|
|
|
Address bind_addr,
|
2015-10-15 19:04:15 +02:00
|
|
|
bool dedicated,
|
2015-07-25 07:43:32 +02:00
|
|
|
ChatInterface *iface
|
2010-11-27 00:02:21 +01:00
|
|
|
):
|
2017-09-28 13:47:30 +02:00
|
|
|
m_bind_addr(bind_addr),
|
2012-03-10 14:56:24 +01:00
|
|
|
m_path_world(path_world),
|
2012-03-11 13:54:23 +01:00
|
|
|
m_gamespec(gamespec),
|
2012-03-15 14:20:20 +01:00
|
|
|
m_simple_singleplayer_mode(simple_singleplayer_mode),
|
2015-10-15 19:04:15 +02:00
|
|
|
m_dedicated(dedicated),
|
2012-03-11 19:45:43 +01:00
|
|
|
m_async_fatal_error(""),
|
2017-08-24 08:28:54 +02:00
|
|
|
m_con(std::make_shared<con::Connection>(PROTOCOL_ID,
|
2014-01-31 00:24:00 +01:00
|
|
|
512,
|
|
|
|
CONNECTION_TIMEOUT,
|
2017-09-28 13:47:30 +02:00
|
|
|
m_bind_addr.isIPv6(),
|
2017-08-24 08:28:54 +02:00
|
|
|
this)),
|
2012-01-12 06:10:39 +01:00
|
|
|
m_itemdef(createItemDefManager()),
|
2011-11-16 13:08:31 +01:00
|
|
|
m_nodedef(createNodeDefManager()),
|
2011-11-17 01:28:46 +01:00
|
|
|
m_craftdef(createCraftDefManager()),
|
2011-01-28 00:38:16 +01:00
|
|
|
m_uptime(0),
|
2017-08-24 08:28:54 +02:00
|
|
|
m_clients(m_con),
|
2017-09-26 00:11:20 +02:00
|
|
|
m_admin_chat(iface),
|
|
|
|
m_modchannel_mgr(new ModChannelMgr())
|
2010-11-27 00:02:21 +01:00
|
|
|
{
|
2014-01-06 23:50:45 +01:00
|
|
|
m_lag = g_settings->getFloat("dedicated_server_step");
|
2011-11-11 18:33:17 +01:00
|
|
|
|
2018-06-13 21:58:34 +02:00
|
|
|
if (m_path_world.empty())
|
2012-03-11 19:45:43 +01:00
|
|
|
throw ServerError("Supplied empty world path");
|
2012-11-26 03:16:48 +01:00
|
|
|
|
2018-06-13 21:58:34 +02:00
|
|
|
if (!gamespec.isValid())
|
2012-03-11 13:54:23 +01:00
|
|
|
throw ServerError("Supplied invalid gamespec");
|
2018-06-13 21:58:34 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
Server::~Server()
|
|
|
|
{
|
|
|
|
|
|
|
|
// Send shutdown message
|
|
|
|
SendChatMessage(PEER_ID_INEXISTENT, ChatMessage(CHATMESSAGE_TYPE_ANNOUNCE,
|
|
|
|
L"*** Server shutting down"));
|
|
|
|
|
|
|
|
if (m_env) {
|
|
|
|
MutexAutoLock envlock(m_env_mutex);
|
|
|
|
|
|
|
|
infostream << "Server: Saving players" << std::endl;
|
|
|
|
m_env->saveLoadedPlayers();
|
2012-11-26 03:16:48 +01:00
|
|
|
|
2018-06-13 21:58:34 +02:00
|
|
|
infostream << "Server: Kicking players" << std::endl;
|
|
|
|
std::string kick_msg;
|
|
|
|
bool reconnect = false;
|
|
|
|
if (isShutdownRequested()) {
|
|
|
|
reconnect = m_shutdown_state.should_reconnect;
|
|
|
|
kick_msg = m_shutdown_state.message;
|
|
|
|
}
|
|
|
|
if (kick_msg.empty()) {
|
|
|
|
kick_msg = g_settings->get("kick_msg_shutdown");
|
|
|
|
}
|
2019-02-03 09:11:45 +01:00
|
|
|
m_env->saveLoadedPlayers(true);
|
2018-06-13 21:58:34 +02:00
|
|
|
m_env->kickAllPlayers(SERVER_ACCESSDENIED_SHUTDOWN,
|
|
|
|
kick_msg, reconnect);
|
|
|
|
}
|
2018-11-12 15:34:01 +01:00
|
|
|
|
2018-07-28 12:58:02 +02:00
|
|
|
actionstream << "Server: Shutting down" << std::endl;
|
2018-06-13 21:58:34 +02:00
|
|
|
|
|
|
|
// Do this before stopping the server in case mapgen callbacks need to access
|
|
|
|
// server-controlled resources (like ModStorages). Also do them before
|
|
|
|
// shutdown callbacks since they may modify state that is finalized in a
|
|
|
|
// callback.
|
|
|
|
if (m_emerge)
|
|
|
|
m_emerge->stopThreads();
|
|
|
|
|
|
|
|
if (m_env) {
|
|
|
|
MutexAutoLock envlock(m_env_mutex);
|
|
|
|
|
|
|
|
// Execute script shutdown hooks
|
|
|
|
infostream << "Executing shutdown hooks" << std::endl;
|
|
|
|
m_script->on_shutdown();
|
|
|
|
|
|
|
|
infostream << "Server: Saving environment metadata" << std::endl;
|
|
|
|
m_env->saveMeta();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Stop threads
|
|
|
|
if (m_thread) {
|
|
|
|
stop();
|
|
|
|
delete m_thread;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Delete things in the reverse order of creation
|
|
|
|
delete m_emerge;
|
|
|
|
delete m_env;
|
|
|
|
delete m_rollback;
|
|
|
|
delete m_banmanager;
|
|
|
|
delete m_itemdef;
|
|
|
|
delete m_nodedef;
|
|
|
|
delete m_craftdef;
|
|
|
|
|
|
|
|
// Deinitialize scripting
|
|
|
|
infostream << "Server: Deinitializing scripting" << std::endl;
|
|
|
|
delete m_script;
|
|
|
|
|
|
|
|
// Delete detached inventories
|
|
|
|
for (auto &detached_inventory : m_detached_inventories) {
|
|
|
|
delete detached_inventory.second;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Server::init()
|
|
|
|
{
|
|
|
|
infostream << "Server created for gameid \"" << m_gamespec.id << "\"";
|
|
|
|
if (m_simple_singleplayer_mode)
|
|
|
|
infostream << " in simple singleplayer mode" << std::endl;
|
2012-03-15 14:20:20 +01:00
|
|
|
else
|
2018-06-13 21:58:34 +02:00
|
|
|
infostream << std::endl;
|
|
|
|
infostream << "- world: " << m_path_world << std::endl;
|
|
|
|
infostream << "- game: " << m_gamespec.path << std::endl;
|
2012-03-11 03:15:45 +01:00
|
|
|
|
2015-01-26 12:44:49 +01:00
|
|
|
// Create world if it doesn't exist
|
2018-06-13 21:58:34 +02:00
|
|
|
if (!loadGameConfAndInitWorld(m_path_world, m_gamespec))
|
2015-01-26 12:44:49 +01:00
|
|
|
throw ServerError("Failed to initialize world");
|
2013-08-11 04:09:45 +02:00
|
|
|
|
|
|
|
// Create server thread
|
|
|
|
m_thread = new ServerThread(this);
|
|
|
|
|
2013-03-24 06:43:38 +01:00
|
|
|
// Create emerge manager
|
2013-04-06 17:19:59 +02:00
|
|
|
m_emerge = new EmergeManager(this);
|
2013-08-11 04:09:45 +02:00
|
|
|
|
|
|
|
// Create ban manager
|
2014-06-26 02:28:41 +02:00
|
|
|
std::string ban_path = m_path_world + DIR_DELIM "ipban.txt";
|
2013-08-11 04:09:45 +02:00
|
|
|
m_banmanager = new BanManager(ban_path);
|
|
|
|
|
2018-06-13 21:58:34 +02:00
|
|
|
m_modmgr = std::unique_ptr<ServerModManager>(new ServerModManager(m_path_world));
|
2018-03-16 08:41:33 +01:00
|
|
|
std::vector<ModSpec> unsatisfied_mods = m_modmgr->getUnsatisfiedMods();
|
2012-12-08 18:10:54 +01:00
|
|
|
// complain about mods with unsatisfied dependencies
|
2018-03-16 08:41:33 +01:00
|
|
|
if (!m_modmgr->isConsistent()) {
|
|
|
|
m_modmgr->printUnsatisfiedModsError();
|
2012-12-08 18:10:54 +01:00
|
|
|
}
|
|
|
|
|
2015-04-07 12:13:12 +02:00
|
|
|
//lock environment
|
|
|
|
MutexAutoLock envlock(m_env_mutex);
|
2012-03-10 14:56:24 +01:00
|
|
|
|
2014-12-30 18:30:14 +01:00
|
|
|
// Create the Map (loads map_meta.txt, overriding configured mapgen params)
|
2018-06-13 21:58:34 +02:00
|
|
|
ServerMap *servermap = new ServerMap(m_path_world, this, m_emerge);
|
2014-12-30 18:30:14 +01:00
|
|
|
|
2011-11-11 18:33:17 +01:00
|
|
|
// Initialize scripting
|
2018-06-13 21:58:34 +02:00
|
|
|
infostream << "Server: Initializing Lua" << std::endl;
|
2013-05-25 00:51:02 +02:00
|
|
|
|
2017-01-21 15:02:08 +01:00
|
|
|
m_script = new ServerScripting(this);
|
2013-05-25 00:51:02 +02:00
|
|
|
|
2017-01-27 07:41:10 +01:00
|
|
|
m_script->loadMod(getBuiltinLuaPath() + DIR_DELIM "init.lua", BUILTIN_MOD_NAME);
|
2014-04-27 23:55:49 +02:00
|
|
|
|
2018-03-16 08:41:33 +01:00
|
|
|
m_modmgr->loadMods(m_script);
|
2012-11-26 03:16:48 +01:00
|
|
|
|
2012-01-02 12:31:50 +01:00
|
|
|
// Read Textures and calculate sha1 sums
|
2012-03-25 10:50:29 +02:00
|
|
|
fillMediaCache();
|
2012-01-02 12:31:50 +01:00
|
|
|
|
2012-01-12 06:10:39 +01:00
|
|
|
// Apply item aliases in the node definition manager
|
|
|
|
m_nodedef->updateAliases(m_itemdef);
|
|
|
|
|
2015-03-30 12:55:29 +02:00
|
|
|
// Apply texture overrides from texturepack/override.txt
|
2017-06-23 02:51:57 +02:00
|
|
|
std::vector<std::string> paths;
|
|
|
|
fs::GetRecursiveDirs(paths, g_settings->get("texture_path"));
|
|
|
|
fs::GetRecursiveDirs(paths, m_gamespec.path + DIR_DELIM + "textures");
|
|
|
|
for (const std::string &path : paths)
|
2017-09-13 22:03:18 +02:00
|
|
|
m_nodedef->applyTextureOverrides(path + DIR_DELIM + "override.txt");
|
2015-03-30 12:55:29 +02:00
|
|
|
|
2015-01-04 22:32:31 +01:00
|
|
|
m_nodedef->setNodeRegistrationStatus(true);
|
|
|
|
|
2014-10-08 21:28:14 +02:00
|
|
|
// Perform pending node name resolutions
|
2015-04-16 10:12:26 +02:00
|
|
|
m_nodedef->runNodeResolveCallbacks();
|
2014-10-08 21:28:14 +02:00
|
|
|
|
Nodebox: Allow nodeboxes to "connect"
We introduce a new nodebox type "connected", and allow these nodes to
have optional nodeboxes that connect it to other connecting nodeboxes.
This is all done at scenedraw time in the client. The client will
inspect the surrounding nodes and if they are to be connected to,
it will draw the appropriate connecting nodeboxes to make those
connections.
In the node_box definition, we have to specify separate nodeboxes for
each valid connection. This allows us to make nodes that connect only
horizontally (the common case) by providing optional nodeboxes for +x,
-x, +z, -z directions. Or this allows us to make wires that can connect
up and down, by providing nodeboxes that connect it up and down (+y,
-y) as well.
The optional nodeboxes can be arrays. They are named "connect_top,
"connect_bottom", "connect_front", "connect_left", "connect_back" and
"connect_right". Here, "front" means the south facing side of the node
that has facedir = 0.
Additionally, a "fixed" nodebox list present will always be drawn,
so one can make a central post, for instance. This "fixed" nodebox
can be omitted, or it can be an array of nodeboxes.
Collision boxes are also updated in exactly the same fashion, which
allows you to walk over the upper extremities of the individual
node boxes, or stand really close to them. You can also walk up
node noxes that are small in height, all as expected, and unlike the
NDT_FENCELIKE nodes.
I've posted a screenshot demonstrating the flexibility at
http://i.imgur.com/zaJq8jo.png
In the screenshot, all connecting nodes are of this new subtype.
Transparent textures render incorrectly, Which I don't think is
related to this text, as other nodeboxes also have issues with this.
A protocol bump is performed in order to be able to send older clients
a nodeblock that is usable for them. In order to avoid abuse of users
we send older clients a "full-size" node, so that it's impossible for
them to try and walk through a fence or wall that's created in this
fashion. This was tested with a pre-bump client connected against a
server running the new protocol.
These nodes connect to other nodes, and you can select which ones
those are by specifying node names (or group names) in the
connects_to string array:
connects_to = { "group:fence", "default:wood" }
By default, nodes do not connect to anything, allowing you to create
nodes that always have to be paired in order to connect. lua_api.txt
is updated to reflect the extension to the node_box API.
Example lua code needed to generate these nodes can be found here:
https://gist.github.com/sofar/b381c8c192c8e53e6062
2016-02-25 09:16:31 +01:00
|
|
|
// unmap node names for connected nodeboxes
|
|
|
|
m_nodedef->mapNodeboxConnections();
|
|
|
|
|
2015-04-04 01:33:34 +02:00
|
|
|
// init the recipe hashes to speed up crafting
|
|
|
|
m_craftdef->initHashes(this);
|
|
|
|
|
2013-01-23 04:32:30 +01:00
|
|
|
// Initialize Environment
|
2014-05-30 22:04:07 +02:00
|
|
|
m_env = new ServerEnvironment(servermap, m_script, this, m_path_world);
|
2014-01-31 00:24:00 +01:00
|
|
|
|
|
|
|
m_clients.setEnv(m_env);
|
|
|
|
|
2016-06-25 00:15:56 +02:00
|
|
|
if (!servermap->settings_mgr.makeMapgenParams())
|
|
|
|
FATAL_ERROR("Couldn't create any mapgen type");
|
|
|
|
|
2013-06-27 23:06:52 +02:00
|
|
|
// Initialize mapgens
|
2016-06-25 00:15:56 +02:00
|
|
|
m_emerge->initMapgens(servermap->getMapgenParams());
|
2012-11-26 03:16:48 +01:00
|
|
|
|
2018-03-16 09:12:39 +01:00
|
|
|
if (g_settings->getBool("enable_rollback_recording")) {
|
2015-02-17 20:09:36 +01:00
|
|
|
// Create rollback manager
|
|
|
|
m_rollback = new RollbackManager(m_path_world, this);
|
|
|
|
}
|
|
|
|
|
2011-11-12 01:25:30 +01:00
|
|
|
// Give environment reference to scripting api
|
2013-05-25 00:51:02 +02:00
|
|
|
m_script->initializeEnvironment(m_env);
|
2012-11-26 03:16:48 +01:00
|
|
|
|
2011-05-22 16:00:09 +02:00
|
|
|
// Register us to receive map edit events
|
2013-01-06 20:40:24 +01:00
|
|
|
servermap->addEventReceiver(this);
|
2011-02-23 01:49:57 +01:00
|
|
|
|
2018-03-16 08:52:42 +01:00
|
|
|
m_env->loadMeta();
|
2011-05-22 16:00:09 +02:00
|
|
|
|
2013-02-24 15:39:07 +01:00
|
|
|
m_liquid_transform_every = g_settings->getFloat("liquid_update");
|
2016-10-04 18:17:12 +02:00
|
|
|
m_max_chatmessage_length = g_settings->getU16("chat_message_max_size");
|
2018-06-20 22:36:08 +02:00
|
|
|
m_csm_restriction_flags = g_settings->getU64("csm_restriction_flags");
|
|
|
|
m_csm_restriction_noderange = g_settings->getU32("csm_restriction_noderange");
|
2010-11-27 00:02:21 +01:00
|
|
|
}
|
|
|
|
|
2017-09-28 13:47:30 +02:00
|
|
|
void Server::start()
|
2010-11-27 00:02:21 +01:00
|
|
|
{
|
2017-09-28 13:47:30 +02:00
|
|
|
infostream << "Starting server on " << m_bind_addr.serializeString()
|
|
|
|
<< "..." << std::endl;
|
2012-03-11 19:45:43 +01:00
|
|
|
|
2010-11-27 00:02:21 +01:00
|
|
|
// Stop thread if already running
|
2015-04-07 12:13:12 +02:00
|
|
|
m_thread->stop();
|
2012-11-26 03:16:48 +01:00
|
|
|
|
2010-11-27 00:02:21 +01:00
|
|
|
// Initialize connection
|
2017-08-24 08:28:54 +02:00
|
|
|
m_con->SetTimeoutMs(30);
|
2017-09-28 13:47:30 +02:00
|
|
|
m_con->Serve(m_bind_addr);
|
2010-11-27 00:02:21 +01:00
|
|
|
|
|
|
|
// Start thread
|
2015-04-07 12:13:12 +02:00
|
|
|
m_thread->start();
|
2012-11-26 03:16:48 +01:00
|
|
|
|
2012-03-11 03:15:45 +01:00
|
|
|
// ASCII art for the win!
|
2018-04-23 19:27:00 +02:00
|
|
|
std::cerr
|
2017-09-28 13:47:30 +02:00
|
|
|
<< " .__ __ __ " << std::endl
|
|
|
|
<< " _____ |__| ____ _____/ |_ ____ _______/ |_ " << std::endl
|
|
|
|
<< " / \\| |/ \\_/ __ \\ __\\/ __ \\ / ___/\\ __\\" << std::endl
|
|
|
|
<< "| Y Y \\ | | \\ ___/| | \\ ___/ \\___ \\ | | " << std::endl
|
|
|
|
<< "|__|_| /__|___| /\\___ >__| \\___ >____ > |__| " << std::endl
|
|
|
|
<< " \\/ \\/ \\/ \\/ \\/ " << std::endl;
|
|
|
|
actionstream << "World at [" << m_path_world << "]" << std::endl;
|
|
|
|
actionstream << "Server for gameid=\"" << m_gamespec.id
|
|
|
|
<< "\" listening on " << m_bind_addr.serializeString() << ":"
|
|
|
|
<< m_bind_addr.getPort() << "." << std::endl;
|
2010-11-27 00:02:21 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void Server::stop()
|
|
|
|
{
|
2011-10-16 13:57:53 +02:00
|
|
|
infostream<<"Server: Stopping and waiting threads"<<std::endl;
|
2011-02-21 15:10:36 +01:00
|
|
|
|
2010-11-27 00:02:21 +01:00
|
|
|
// Stop threads (set run=false first so both start stopping)
|
2015-04-07 12:13:12 +02:00
|
|
|
m_thread->stop();
|
2013-02-14 04:43:15 +01:00
|
|
|
//m_emergethread.setRun(false);
|
2015-04-07 12:13:12 +02:00
|
|
|
m_thread->wait();
|
2013-02-14 04:43:15 +01:00
|
|
|
//m_emergethread.stop();
|
2012-11-26 03:16:48 +01:00
|
|
|
|
2011-10-16 13:57:53 +02:00
|
|
|
infostream<<"Server: Threads stopped"<<std::endl;
|
2010-11-27 00:02:21 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void Server::step(float dtime)
|
|
|
|
{
|
|
|
|
// Limit a bit
|
2015-10-29 19:48:10 +01:00
|
|
|
if (dtime > 2.0)
|
2010-11-27 00:02:21 +01:00
|
|
|
dtime = 2.0;
|
|
|
|
{
|
2015-04-07 12:13:12 +02:00
|
|
|
MutexAutoLock lock(m_step_dtime_mutex);
|
2010-11-27 00:02:21 +01:00
|
|
|
m_step_dtime += dtime;
|
|
|
|
}
|
2015-03-17 02:17:27 +01:00
|
|
|
// Throw if fatal error occurred in thread
|
2012-03-11 19:45:43 +01:00
|
|
|
std::string async_err = m_async_fatal_error.get();
|
2015-10-29 19:48:10 +01:00
|
|
|
if (!async_err.empty()) {
|
|
|
|
if (!m_simple_singleplayer_mode) {
|
2015-07-17 16:40:41 +02:00
|
|
|
m_env->kickAllPlayers(SERVER_ACCESSDENIED_CRASH,
|
|
|
|
g_settings->get("kick_msg_crash"),
|
|
|
|
g_settings->getBool("ask_reconnect_on_crash"));
|
2015-03-17 09:13:12 +01:00
|
|
|
}
|
2017-01-15 13:36:53 +01:00
|
|
|
throw ServerError("AsyncErr: " + async_err);
|
2012-03-11 19:45:43 +01:00
|
|
|
}
|
2010-11-27 00:02:21 +01:00
|
|
|
}
|
|
|
|
|
2014-01-06 20:05:28 +01:00
|
|
|
void Server::AsyncRunStep(bool initial_step)
|
2010-11-27 00:02:21 +01:00
|
|
|
{
|
2011-10-16 20:16:44 +02:00
|
|
|
g_profiler->add("Server::AsyncRunStep (num)", 1);
|
2012-11-26 03:16:48 +01:00
|
|
|
|
2010-11-27 00:02:21 +01:00
|
|
|
float dtime;
|
|
|
|
{
|
2015-04-07 12:13:12 +02:00
|
|
|
MutexAutoLock lock1(m_step_dtime_mutex);
|
2010-11-27 00:02:21 +01:00
|
|
|
dtime = m_step_dtime;
|
|
|
|
}
|
2012-11-26 03:16:48 +01:00
|
|
|
|
2011-05-30 23:15:43 +02:00
|
|
|
{
|
|
|
|
// Send blocks to clients
|
|
|
|
SendBlocks(dtime);
|
|
|
|
}
|
2012-11-26 03:16:48 +01:00
|
|
|
|
2017-08-14 01:06:12 +02:00
|
|
|
if((dtime < 0.001) && !initial_step)
|
2010-12-13 20:32:35 +01:00
|
|
|
return;
|
2012-11-26 03:16:48 +01:00
|
|
|
|
2011-10-16 20:16:44 +02:00
|
|
|
g_profiler->add("Server::AsyncRunStep with dtime (num)", 1);
|
2011-05-31 10:59:51 +02:00
|
|
|
|
2011-10-16 13:57:53 +02:00
|
|
|
//infostream<<"Server steps "<<dtime<<std::endl;
|
|
|
|
//infostream<<"Server::AsyncRunStep(): dtime="<<dtime<<std::endl;
|
2012-11-26 03:16:48 +01:00
|
|
|
|
2010-12-13 20:32:35 +01:00
|
|
|
{
|
2015-04-07 12:13:12 +02:00
|
|
|
MutexAutoLock lock1(m_step_dtime_mutex);
|
2010-12-19 15:51:45 +01:00
|
|
|
m_step_dtime -= dtime;
|
2010-12-13 20:32:35 +01:00
|
|
|
}
|
2010-12-24 16:08:50 +01:00
|
|
|
|
|
|
|
/*
|
|
|
|
Update uptime
|
|
|
|
*/
|
|
|
|
{
|
|
|
|
m_uptime.set(m_uptime.get() + dtime);
|
|
|
|
}
|
2012-11-26 03:16:48 +01:00
|
|
|
|
2014-01-31 00:24:00 +01:00
|
|
|
handlePeerChanges();
|
2011-08-22 20:27:11 +02:00
|
|
|
|
2010-12-20 13:04:31 +01:00
|
|
|
/*
|
2012-03-16 15:34:30 +01:00
|
|
|
Update time of day and overall game time
|
2010-12-20 13:04:31 +01:00
|
|
|
*/
|
2015-03-04 11:46:31 +01:00
|
|
|
m_env->setTimeOfDaySpeed(g_settings->getFloat("time_speed"));
|
2011-05-22 16:00:09 +02:00
|
|
|
|
2015-03-04 11:46:31 +01:00
|
|
|
/*
|
|
|
|
Send to clients at constant intervals
|
|
|
|
*/
|
2010-12-20 13:04:31 +01:00
|
|
|
|
2015-03-04 11:46:31 +01:00
|
|
|
m_time_of_day_send_timer -= dtime;
|
|
|
|
if(m_time_of_day_send_timer < 0.0) {
|
|
|
|
m_time_of_day_send_timer = g_settings->getFloat("time_send_interval");
|
|
|
|
u16 time = m_env->getTimeOfDay();
|
|
|
|
float time_speed = g_settings->getFloat("time_speed");
|
|
|
|
SendTimeOfDay(PEER_ID_INEXISTENT, time, time_speed);
|
2010-12-20 13:04:31 +01:00
|
|
|
}
|
2010-12-13 20:32:35 +01:00
|
|
|
|
2010-11-27 00:02:21 +01:00
|
|
|
{
|
2015-04-07 12:13:12 +02:00
|
|
|
MutexAutoLock lock(m_env_mutex);
|
2013-08-03 22:16:37 +02:00
|
|
|
// Figure out and report maximum lag to environment
|
|
|
|
float max_lag = m_env->getMaxLagEstimate();
|
|
|
|
max_lag *= 0.9998; // Decrease slowly (about half per 5 minutes)
|
|
|
|
if(dtime > max_lag){
|
|
|
|
if(dtime > 0.1 && dtime > max_lag * 2.0)
|
|
|
|
infostream<<"Server: Maximum lag peaked to "<<dtime
|
|
|
|
<<" s"<<std::endl;
|
|
|
|
max_lag = dtime;
|
|
|
|
}
|
|
|
|
m_env->reportMaxLagEstimate(max_lag);
|
2011-06-26 23:27:17 +02:00
|
|
|
// Step environment
|
2011-10-16 20:16:44 +02:00
|
|
|
ScopeProfiler sp(g_profiler, "SEnv step");
|
2011-10-16 21:39:35 +02:00
|
|
|
ScopeProfiler sp2(g_profiler, "SEnv step avg", SPT_AVG);
|
2011-11-11 18:33:17 +01:00
|
|
|
m_env->step(dtime);
|
2010-11-27 00:02:21 +01:00
|
|
|
}
|
2012-11-26 03:16:48 +01:00
|
|
|
|
2014-12-12 09:06:36 +01:00
|
|
|
static const float map_timer_and_unload_dtime = 2.92;
|
2011-06-26 23:27:17 +02:00
|
|
|
if(m_map_timer_and_unload_interval.step(dtime, map_timer_and_unload_dtime))
|
|
|
|
{
|
2015-04-07 12:13:12 +02:00
|
|
|
MutexAutoLock lock(m_env_mutex);
|
2011-06-26 23:27:17 +02:00
|
|
|
// Run Map's timers and unload unused data
|
2011-10-12 12:53:38 +02:00
|
|
|
ScopeProfiler sp(g_profiler, "Server: map timer and unload");
|
2011-11-11 18:33:17 +01:00
|
|
|
m_env->getMap().timerUpdate(map_timer_and_unload_dtime,
|
2015-08-10 22:24:47 +02:00
|
|
|
g_settings->getFloat("server_unload_unused_data_timeout"),
|
2015-10-04 08:50:04 +02:00
|
|
|
U32_MAX);
|
2011-06-26 23:27:17 +02:00
|
|
|
}
|
2012-11-26 03:16:48 +01:00
|
|
|
|
2015-07-25 07:43:32 +02:00
|
|
|
/*
|
|
|
|
Listen to the admin chat, if available
|
|
|
|
*/
|
|
|
|
if (m_admin_chat) {
|
|
|
|
if (!m_admin_chat->command_queue.empty()) {
|
|
|
|
MutexAutoLock lock(m_env_mutex);
|
|
|
|
while (!m_admin_chat->command_queue.empty()) {
|
|
|
|
ChatEvent *evt = m_admin_chat->command_queue.pop_frontNoEx();
|
2015-11-09 06:04:24 +01:00
|
|
|
handleChatInterfaceEvent(evt);
|
2015-07-25 07:43:32 +02:00
|
|
|
delete evt;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
m_admin_chat->outgoing_queue.push_back(
|
|
|
|
new ChatEventTimeInfo(m_env->getGameTime(), m_env->getTimeOfDay()));
|
|
|
|
}
|
|
|
|
|
2010-11-27 00:02:21 +01:00
|
|
|
/*
|
|
|
|
Do background stuff
|
|
|
|
*/
|
2011-11-26 13:03:56 +01:00
|
|
|
|
|
|
|
/* Transform liquids */
|
2011-01-17 20:15:31 +01:00
|
|
|
m_liquid_transform_timer += dtime;
|
2013-02-24 15:39:07 +01:00
|
|
|
if(m_liquid_transform_timer >= m_liquid_transform_every)
|
2011-01-17 13:57:37 +01:00
|
|
|
{
|
2013-02-24 15:39:07 +01:00
|
|
|
m_liquid_transform_timer -= m_liquid_transform_every;
|
2012-11-26 03:16:48 +01:00
|
|
|
|
2015-04-07 12:13:12 +02:00
|
|
|
MutexAutoLock lock(m_env_mutex);
|
2011-05-30 23:15:43 +02:00
|
|
|
|
2011-10-12 12:53:38 +02:00
|
|
|
ScopeProfiler sp(g_profiler, "Server: liquid transform");
|
2011-05-30 23:15:43 +02:00
|
|
|
|
2012-12-20 18:19:49 +01:00
|
|
|
std::map<v3s16, MapBlock*> modified_blocks;
|
2017-04-20 08:10:39 +02:00
|
|
|
m_env->getMap().transformLiquids(modified_blocks, m_env);
|
2017-08-14 01:06:12 +02:00
|
|
|
|
2011-01-17 20:15:31 +01:00
|
|
|
/*
|
|
|
|
Set the modified blocks unsent for all the clients
|
|
|
|
*/
|
2018-03-08 22:58:43 +01:00
|
|
|
if (!modified_blocks.empty()) {
|
2014-01-31 00:24:00 +01:00
|
|
|
SetBlocksNotSent(modified_blocks);
|
2010-11-27 00:02:21 +01:00
|
|
|
}
|
|
|
|
}
|
2014-01-31 00:24:00 +01:00
|
|
|
m_clients.step(dtime);
|
2013-02-21 23:00:44 +01:00
|
|
|
|
2014-01-06 23:50:45 +01:00
|
|
|
m_lag += (m_lag > dtime ? -1 : 1) * dtime/100;
|
2013-02-21 23:00:44 +01:00
|
|
|
#if USE_CURL
|
|
|
|
// send masterserver announce
|
|
|
|
{
|
|
|
|
float &counter = m_masterserver_timer;
|
2015-10-15 19:04:15 +02:00
|
|
|
if (!isSingleplayer() && (!counter || counter >= 300.0) &&
|
|
|
|
g_settings->getBool("server_announce")) {
|
|
|
|
ServerList::sendAnnounce(counter ? ServerList::AA_UPDATE :
|
|
|
|
ServerList::AA_START,
|
2015-01-21 20:32:12 +01:00
|
|
|
m_bind_addr.getPort(),
|
2014-06-20 06:59:39 +02:00
|
|
|
m_clients.getPlayerNames(),
|
|
|
|
m_uptime.get(),
|
|
|
|
m_env->getGameTime(),
|
|
|
|
m_lag,
|
|
|
|
m_gamespec.id,
|
2016-06-25 00:15:56 +02:00
|
|
|
Mapgen::getMapgenName(m_emerge->mgparams->mgtype),
|
2018-03-16 08:41:33 +01:00
|
|
|
m_modmgr->getMods(),
|
2015-10-15 19:04:15 +02:00
|
|
|
m_dedicated);
|
2013-02-21 23:00:44 +01:00
|
|
|
counter = 0.01;
|
|
|
|
}
|
|
|
|
counter += dtime;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2010-12-18 12:10:37 +01:00
|
|
|
/*
|
2011-02-20 23:45:14 +01:00
|
|
|
Check added and deleted active objects
|
2010-12-18 12:10:37 +01:00
|
|
|
*/
|
|
|
|
{
|
2011-10-16 13:57:53 +02:00
|
|
|
//infostream<<"Server: Checking added and deleted active objects"<<std::endl;
|
2015-04-07 12:13:12 +02:00
|
|
|
MutexAutoLock envlock(m_env_mutex);
|
2011-05-30 23:15:43 +02:00
|
|
|
|
2015-04-07 12:13:12 +02:00
|
|
|
m_clients.lock();
|
2017-08-14 01:06:12 +02:00
|
|
|
const RemoteClientMap &clients = m_clients.getClientList();
|
2011-10-16 20:16:44 +02:00
|
|
|
ScopeProfiler sp(g_profiler, "Server: checking added and deleted objs");
|
2011-05-30 23:15:43 +02:00
|
|
|
|
2011-02-23 01:49:57 +01:00
|
|
|
// Radius inside which objects are active
|
2017-06-11 13:58:43 +02:00
|
|
|
static thread_local const s16 radius =
|
2015-09-02 21:21:55 +02:00
|
|
|
g_settings->getS16("active_object_send_range_blocks") * MAP_BLOCKSIZE;
|
|
|
|
|
|
|
|
// Radius inside which players are active
|
2017-06-11 13:58:43 +02:00
|
|
|
static thread_local const bool is_transfer_limited =
|
2015-09-02 21:21:55 +02:00
|
|
|
g_settings->exists("unlimited_player_transfer_distance") &&
|
|
|
|
!g_settings->getBool("unlimited_player_transfer_distance");
|
2017-06-11 13:58:43 +02:00
|
|
|
static thread_local const s16 player_transfer_dist =
|
|
|
|
g_settings->getS16("player_transfer_distance") * MAP_BLOCKSIZE;
|
2015-09-02 21:21:55 +02:00
|
|
|
s16 player_radius = player_transfer_dist;
|
|
|
|
if (player_radius == 0 && is_transfer_limited)
|
2014-11-08 14:35:55 +01:00
|
|
|
player_radius = radius;
|
|
|
|
|
2017-08-14 01:06:12 +02:00
|
|
|
for (const auto &client_it : clients) {
|
|
|
|
RemoteClient *client = client_it.second;
|
2011-12-02 00:18:25 +01:00
|
|
|
|
|
|
|
// If definitions and textures have not been sent, don't
|
|
|
|
// send objects either
|
2014-06-28 08:02:38 +02:00
|
|
|
if (client->getState() < CS_DefinitionsSent)
|
2011-12-02 00:18:25 +01:00
|
|
|
continue;
|
|
|
|
|
2016-10-08 16:31:22 +02:00
|
|
|
RemotePlayer *player = m_env->getPlayer(client->peer_id);
|
2017-08-14 01:06:12 +02:00
|
|
|
if (!player) {
|
2011-06-02 19:09:30 +02:00
|
|
|
// This can happen if the client timeouts somehow
|
2011-02-23 01:49:57 +01:00
|
|
|
continue;
|
2011-04-07 23:47:14 +02:00
|
|
|
}
|
2011-02-20 23:45:14 +01:00
|
|
|
|
2016-10-30 14:53:26 +01:00
|
|
|
PlayerSAO *playersao = player->getPlayerSAO();
|
2017-08-14 01:06:12 +02:00
|
|
|
if (!playersao)
|
2016-10-30 14:53:26 +01:00
|
|
|
continue;
|
|
|
|
|
2016-11-30 09:13:14 +01:00
|
|
|
s16 my_radius = MYMIN(radius, playersao->getWantedRange() * MAP_BLOCKSIZE);
|
|
|
|
if (my_radius <= 0) my_radius = radius;
|
|
|
|
//infostream << "Server: Active Radius " << my_radius << std::endl;
|
|
|
|
|
2015-08-05 22:29:47 +02:00
|
|
|
std::queue<u16> removed_objects;
|
|
|
|
std::queue<u16> added_objects;
|
2016-11-30 09:13:14 +01:00
|
|
|
m_env->getRemovedActiveObjects(playersao, my_radius, player_radius,
|
2011-02-20 23:45:14 +01:00
|
|
|
client->m_known_objects, removed_objects);
|
2016-11-30 09:13:14 +01:00
|
|
|
m_env->getAddedActiveObjects(playersao, my_radius, player_radius,
|
2011-02-20 23:45:14 +01:00
|
|
|
client->m_known_objects, added_objects);
|
2012-11-26 03:16:48 +01:00
|
|
|
|
2011-02-20 23:45:14 +01:00
|
|
|
// Ignore if nothing happened
|
2015-08-05 22:29:47 +02:00
|
|
|
if (removed_objects.empty() && added_objects.empty()) {
|
2011-02-20 23:45:14 +01:00
|
|
|
continue;
|
2011-04-07 23:47:14 +02:00
|
|
|
}
|
2012-11-26 03:16:48 +01:00
|
|
|
|
2011-02-20 23:45:14 +01:00
|
|
|
std::string data_buffer;
|
2010-12-18 12:10:37 +01:00
|
|
|
|
2011-02-20 23:45:14 +01:00
|
|
|
char buf[4];
|
2012-11-26 03:16:48 +01:00
|
|
|
|
2011-02-20 23:45:14 +01:00
|
|
|
// Handle removed objects
|
|
|
|
writeU16((u8*)buf, removed_objects.size());
|
|
|
|
data_buffer.append(buf, 2);
|
2015-08-05 22:29:47 +02:00
|
|
|
while (!removed_objects.empty()) {
|
2011-02-20 23:45:14 +01:00
|
|
|
// Get object
|
2015-08-05 22:29:47 +02:00
|
|
|
u16 id = removed_objects.front();
|
2011-11-11 18:33:17 +01:00
|
|
|
ServerActiveObject* obj = m_env->getActiveObject(id);
|
2010-12-18 12:10:37 +01:00
|
|
|
|
2011-02-20 23:45:14 +01:00
|
|
|
// Add to data buffer for sending
|
2012-12-20 18:19:49 +01:00
|
|
|
writeU16((u8*)buf, id);
|
2011-02-20 23:45:14 +01:00
|
|
|
data_buffer.append(buf, 2);
|
2012-11-26 03:16:48 +01:00
|
|
|
|
2011-02-20 23:45:14 +01:00
|
|
|
// Remove from known objects
|
2012-12-20 18:19:49 +01:00
|
|
|
client->m_known_objects.erase(id);
|
2010-12-18 12:10:37 +01:00
|
|
|
|
2011-02-20 23:45:14 +01:00
|
|
|
if(obj && obj->m_known_by_count > 0)
|
|
|
|
obj->m_known_by_count--;
|
2015-08-05 22:29:47 +02:00
|
|
|
removed_objects.pop();
|
2011-02-20 23:45:14 +01:00
|
|
|
}
|
2010-12-18 12:10:37 +01:00
|
|
|
|
2011-02-20 23:45:14 +01:00
|
|
|
// Handle added objects
|
|
|
|
writeU16((u8*)buf, added_objects.size());
|
|
|
|
data_buffer.append(buf, 2);
|
2015-08-05 22:29:47 +02:00
|
|
|
while (!added_objects.empty()) {
|
2011-02-20 23:45:14 +01:00
|
|
|
// Get object
|
2015-08-05 22:29:47 +02:00
|
|
|
u16 id = added_objects.front();
|
2011-11-11 18:33:17 +01:00
|
|
|
ServerActiveObject* obj = m_env->getActiveObject(id);
|
2012-11-26 03:16:48 +01:00
|
|
|
|
2011-02-20 23:45:14 +01:00
|
|
|
// Get object type
|
|
|
|
u8 type = ACTIVEOBJECT_TYPE_INVALID;
|
2017-08-14 01:06:12 +02:00
|
|
|
if (!obj)
|
|
|
|
warningstream << FUNCTION_NAME << ": NULL object" << std::endl;
|
2011-02-20 23:45:14 +01:00
|
|
|
else
|
2012-03-29 15:10:11 +02:00
|
|
|
type = obj->getSendType();
|
2011-02-20 23:45:14 +01:00
|
|
|
|
|
|
|
// Add to data buffer for sending
|
|
|
|
writeU16((u8*)buf, id);
|
|
|
|
data_buffer.append(buf, 2);
|
|
|
|
writeU8((u8*)buf, type);
|
|
|
|
data_buffer.append(buf, 1);
|
2012-11-26 03:16:48 +01:00
|
|
|
|
2011-04-11 10:05:40 +02:00
|
|
|
if(obj)
|
|
|
|
data_buffer.append(serializeLongString(
|
2012-11-26 23:47:03 +01:00
|
|
|
obj->getClientInitializationData(client->net_proto_version)));
|
2011-04-11 10:05:40 +02:00
|
|
|
else
|
|
|
|
data_buffer.append(serializeLongString(""));
|
2011-02-21 15:10:36 +01:00
|
|
|
|
2011-02-20 23:45:14 +01:00
|
|
|
// Add to known objects
|
2012-12-20 18:19:49 +01:00
|
|
|
client->m_known_objects.insert(id);
|
2011-02-20 23:45:14 +01:00
|
|
|
|
|
|
|
if(obj)
|
|
|
|
obj->m_known_by_count++;
|
2015-08-05 22:29:47 +02:00
|
|
|
|
|
|
|
added_objects.pop();
|
2010-12-23 21:35:53 +01:00
|
|
|
}
|
2010-12-18 12:10:37 +01:00
|
|
|
|
2015-03-16 12:01:40 +01:00
|
|
|
u32 pktSize = SendActiveObjectRemoveAdd(client->peer_id, data_buffer);
|
[Patch 2/4] Network rework: packet writing, sending and cleanups
NetworkPacket.cpp:
* Remove some deprecated functions, we must use streaming interface
* m_data converted from u8* to std::vector<u8>
* Add an exporter to forge packet to Connection object
* implement operator << std::wstring. n
* implement operator << std::string
* dynamic resize when write packet content.
* fix string writing and performances.
* create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Reliability
* Transmit channel
* Implement putRawString for some ugly char (_INIT packet), and use it.
* Many packet read and write migrated
* Implement oldForgePacket to interface writing with current connection
* fix U8/char/bool writing
* fix string writing and performances.
* add some missing functions
* Use v3s16 read instead of reading x,y,z separately
* Add irr::video::SColor support into packets
* Add some missing handlers
* Add a template function to increase offset
* Throw a serialization error on packet reading (must be improved)
PacketFactories:
* Create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Create ClientCommandFactory, used by server to get useful informations about packet processing (sending).
Client.cpp:
* implement NetworkPacket ::Send interface.
* Move packet handlers to a dedicated file
* Remove Client::Send(SharedBuffer)
Server.cpp:
* implement NetworkPacket ::Send interface.
* Rewrite all packets using NetworkPacket
* Move packet handlers to a dedicated file
* Remove Server::Send(SharedBuffer)
ClientIface.cpp:
* Remove sendToAll(SharedBuffer<u8>)
Connection.hpp rework:
* Remove duplicate include
* Remove duplicate negation
* Remove a useless variable
* Improve code performance by using a m_peers_list instead of scanning m_peers map
* Remove Connection::Send(SharedBuffer)
* Fix useafterfree into NetworkPacket Sending
* Remove unused Connection::sendToAll
Test.cpp:
* Remove dead code
* Update tests to use NetworkPackets
Misc:
* add new wrappers to Send packets in client, using NetworkPacket
* Add NetworkPacket methods for Connection
* coding style fix
* dead code since changes cleanup
* Use v3s16 read instead of reading x,y,z separately in some packets
* Use different files to handle packets received by client and server
* Cleanup: Remove useless includes
ok @Zeno-
Tested by @Zeno- @VanessaE and @nerzhul on running servers
2015-01-16 11:37:49 +01:00
|
|
|
verbosestream << "Server: Sent object remove/add: "
|
|
|
|
<< removed_objects.size() << " removed, "
|
|
|
|
<< added_objects.size() << " added, "
|
2015-03-16 12:01:40 +01:00
|
|
|
<< "packet size is " << pktSize << std::endl;
|
[Patch 2/4] Network rework: packet writing, sending and cleanups
NetworkPacket.cpp:
* Remove some deprecated functions, we must use streaming interface
* m_data converted from u8* to std::vector<u8>
* Add an exporter to forge packet to Connection object
* implement operator << std::wstring. n
* implement operator << std::string
* dynamic resize when write packet content.
* fix string writing and performances.
* create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Reliability
* Transmit channel
* Implement putRawString for some ugly char (_INIT packet), and use it.
* Many packet read and write migrated
* Implement oldForgePacket to interface writing with current connection
* fix U8/char/bool writing
* fix string writing and performances.
* add some missing functions
* Use v3s16 read instead of reading x,y,z separately
* Add irr::video::SColor support into packets
* Add some missing handlers
* Add a template function to increase offset
* Throw a serialization error on packet reading (must be improved)
PacketFactories:
* Create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Create ClientCommandFactory, used by server to get useful informations about packet processing (sending).
Client.cpp:
* implement NetworkPacket ::Send interface.
* Move packet handlers to a dedicated file
* Remove Client::Send(SharedBuffer)
Server.cpp:
* implement NetworkPacket ::Send interface.
* Rewrite all packets using NetworkPacket
* Move packet handlers to a dedicated file
* Remove Server::Send(SharedBuffer)
ClientIface.cpp:
* Remove sendToAll(SharedBuffer<u8>)
Connection.hpp rework:
* Remove duplicate include
* Remove duplicate negation
* Remove a useless variable
* Improve code performance by using a m_peers_list instead of scanning m_peers map
* Remove Connection::Send(SharedBuffer)
* Fix useafterfree into NetworkPacket Sending
* Remove unused Connection::sendToAll
Test.cpp:
* Remove dead code
* Update tests to use NetworkPackets
Misc:
* add new wrappers to Send packets in client, using NetworkPacket
* Add NetworkPacket methods for Connection
* coding style fix
* dead code since changes cleanup
* Use v3s16 read instead of reading x,y,z separately in some packets
* Use different files to handle packets received by client and server
* Cleanup: Remove useless includes
ok @Zeno-
Tested by @Zeno- @VanessaE and @nerzhul on running servers
2015-01-16 11:37:49 +01:00
|
|
|
}
|
2015-04-07 12:13:12 +02:00
|
|
|
m_clients.unlock();
|
2017-02-08 00:15:55 +01:00
|
|
|
|
|
|
|
m_mod_storage_save_timer -= dtime;
|
|
|
|
if (m_mod_storage_save_timer <= 0.0f) {
|
|
|
|
infostream << "Saving registered mod storages." << std::endl;
|
|
|
|
m_mod_storage_save_timer = g_settings->getFloat("server_map_save_interval");
|
2017-06-04 21:00:04 +02:00
|
|
|
for (std::unordered_map<std::string, ModMetadata *>::const_iterator
|
2017-02-08 00:15:55 +01:00
|
|
|
it = m_mod_storages.begin(); it != m_mod_storages.end(); ++it) {
|
|
|
|
if (it->second->isModified()) {
|
|
|
|
it->second->save(getModStoragePath());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2011-02-20 23:45:14 +01:00
|
|
|
}
|
2010-12-18 12:10:37 +01:00
|
|
|
|
2011-02-20 23:45:14 +01:00
|
|
|
/*
|
|
|
|
Send object messages
|
|
|
|
*/
|
|
|
|
{
|
2015-04-07 12:13:12 +02:00
|
|
|
MutexAutoLock envlock(m_env_mutex);
|
2011-12-02 16:19:42 +01:00
|
|
|
ScopeProfiler sp(g_profiler, "Server: sending object messages");
|
2011-05-30 23:15:43 +02:00
|
|
|
|
2011-02-20 23:45:14 +01:00
|
|
|
// Key = object id
|
|
|
|
// Value = data sent by object
|
2017-06-04 21:00:04 +02:00
|
|
|
std::unordered_map<u16, std::vector<ActiveObjectMessage>*> buffered_messages;
|
2011-02-20 23:45:14 +01:00
|
|
|
|
|
|
|
// Get active object messages from environment
|
2015-03-05 11:18:17 +01:00
|
|
|
for(;;) {
|
2011-11-11 18:33:17 +01:00
|
|
|
ActiveObjectMessage aom = m_env->getActiveObjectMessage();
|
2015-03-05 11:18:17 +01:00
|
|
|
if (aom.id == 0)
|
2011-02-20 23:45:14 +01:00
|
|
|
break;
|
2012-11-26 03:16:48 +01:00
|
|
|
|
2017-08-14 00:44:45 +02:00
|
|
|
std::vector<ActiveObjectMessage>* message_list = nullptr;
|
2017-06-04 21:00:04 +02:00
|
|
|
std::unordered_map<u16, std::vector<ActiveObjectMessage>* >::iterator n;
|
2011-02-20 23:45:14 +01:00
|
|
|
n = buffered_messages.find(aom.id);
|
2015-03-05 11:18:17 +01:00
|
|
|
if (n == buffered_messages.end()) {
|
|
|
|
message_list = new std::vector<ActiveObjectMessage>;
|
2012-12-20 18:19:49 +01:00
|
|
|
buffered_messages[aom.id] = message_list;
|
2010-12-18 12:10:37 +01:00
|
|
|
}
|
2015-03-05 11:18:17 +01:00
|
|
|
else {
|
2012-12-20 18:19:49 +01:00
|
|
|
message_list = n->second;
|
2010-12-18 12:10:37 +01:00
|
|
|
}
|
2011-02-20 23:45:14 +01:00
|
|
|
message_list->push_back(aom);
|
|
|
|
}
|
2012-11-26 03:16:48 +01:00
|
|
|
|
2015-04-07 12:13:12 +02:00
|
|
|
m_clients.lock();
|
2017-08-14 00:44:45 +02:00
|
|
|
const RemoteClientMap &clients = m_clients.getClientList();
|
2011-02-20 23:45:14 +01:00
|
|
|
// Route data to every client
|
2017-08-14 00:44:45 +02:00
|
|
|
for (const auto &client_it : clients) {
|
|
|
|
RemoteClient *client = client_it.second;
|
2011-02-20 23:45:14 +01:00
|
|
|
std::string reliable_data;
|
|
|
|
std::string unreliable_data;
|
|
|
|
// Go through all objects in message buffer
|
2017-08-14 00:44:45 +02:00
|
|
|
for (const auto &buffered_message : buffered_messages) {
|
2011-02-20 23:45:14 +01:00
|
|
|
// If object is not known by client, skip it
|
2017-08-14 00:44:45 +02:00
|
|
|
u16 id = buffered_message.first;
|
2015-03-05 11:18:17 +01:00
|
|
|
if (client->m_known_objects.find(id) == client->m_known_objects.end())
|
2011-02-20 23:45:14 +01:00
|
|
|
continue;
|
2015-03-05 11:18:17 +01:00
|
|
|
|
2011-02-20 23:45:14 +01:00
|
|
|
// Get message list of object
|
2017-08-14 00:44:45 +02:00
|
|
|
std::vector<ActiveObjectMessage>* list = buffered_message.second;
|
2011-02-20 23:45:14 +01:00
|
|
|
// Go through every message
|
2017-08-14 00:44:45 +02:00
|
|
|
for (const ActiveObjectMessage &aom : *list) {
|
2011-02-20 23:45:14 +01:00
|
|
|
// Compose the full new data with header
|
|
|
|
std::string new_data;
|
2011-02-21 15:10:36 +01:00
|
|
|
// Add object id
|
|
|
|
char buf[2];
|
|
|
|
writeU16((u8*)&buf[0], aom.id);
|
|
|
|
new_data.append(buf, 2);
|
2011-02-20 23:45:14 +01:00
|
|
|
// Add data
|
2011-02-21 15:10:36 +01:00
|
|
|
new_data += serializeString(aom.datastring);
|
2011-02-20 23:45:14 +01:00
|
|
|
// Add data to buffer
|
2017-08-14 00:44:45 +02:00
|
|
|
if (aom.reliable)
|
2011-02-20 23:45:14 +01:00
|
|
|
reliable_data += new_data;
|
|
|
|
else
|
|
|
|
unreliable_data += new_data;
|
|
|
|
}
|
2010-12-18 12:10:37 +01:00
|
|
|
}
|
|
|
|
/*
|
2011-02-20 23:45:14 +01:00
|
|
|
reliable_data and unreliable_data are now ready.
|
|
|
|
Send them.
|
2010-12-18 12:10:37 +01:00
|
|
|
*/
|
2017-08-14 00:44:45 +02:00
|
|
|
if (!reliable_data.empty()) {
|
2015-03-16 12:01:40 +01:00
|
|
|
SendActiveObjectMessages(client->peer_id, reliable_data);
|
2010-12-18 12:10:37 +01:00
|
|
|
}
|
2011-02-21 15:10:36 +01:00
|
|
|
|
2017-08-14 00:44:45 +02:00
|
|
|
if (!unreliable_data.empty()) {
|
2015-03-17 09:16:39 +01:00
|
|
|
SendActiveObjectMessages(client->peer_id, unreliable_data, false);
|
[Patch 2/4] Network rework: packet writing, sending and cleanups
NetworkPacket.cpp:
* Remove some deprecated functions, we must use streaming interface
* m_data converted from u8* to std::vector<u8>
* Add an exporter to forge packet to Connection object
* implement operator << std::wstring. n
* implement operator << std::string
* dynamic resize when write packet content.
* fix string writing and performances.
* create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Reliability
* Transmit channel
* Implement putRawString for some ugly char (_INIT packet), and use it.
* Many packet read and write migrated
* Implement oldForgePacket to interface writing with current connection
* fix U8/char/bool writing
* fix string writing and performances.
* add some missing functions
* Use v3s16 read instead of reading x,y,z separately
* Add irr::video::SColor support into packets
* Add some missing handlers
* Add a template function to increase offset
* Throw a serialization error on packet reading (must be improved)
PacketFactories:
* Create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Create ClientCommandFactory, used by server to get useful informations about packet processing (sending).
Client.cpp:
* implement NetworkPacket ::Send interface.
* Move packet handlers to a dedicated file
* Remove Client::Send(SharedBuffer)
Server.cpp:
* implement NetworkPacket ::Send interface.
* Rewrite all packets using NetworkPacket
* Move packet handlers to a dedicated file
* Remove Server::Send(SharedBuffer)
ClientIface.cpp:
* Remove sendToAll(SharedBuffer<u8>)
Connection.hpp rework:
* Remove duplicate include
* Remove duplicate negation
* Remove a useless variable
* Improve code performance by using a m_peers_list instead of scanning m_peers map
* Remove Connection::Send(SharedBuffer)
* Fix useafterfree into NetworkPacket Sending
* Remove unused Connection::sendToAll
Test.cpp:
* Remove dead code
* Update tests to use NetworkPackets
Misc:
* add new wrappers to Send packets in client, using NetworkPacket
* Add NetworkPacket methods for Connection
* coding style fix
* dead code since changes cleanup
* Use v3s16 read instead of reading x,y,z separately in some packets
* Use different files to handle packets received by client and server
* Cleanup: Remove useless includes
ok @Zeno-
Tested by @Zeno- @VanessaE and @nerzhul on running servers
2015-01-16 11:37:49 +01:00
|
|
|
}
|
2011-02-20 23:45:14 +01:00
|
|
|
}
|
2015-04-07 12:13:12 +02:00
|
|
|
m_clients.unlock();
|
2011-02-20 23:45:14 +01:00
|
|
|
|
|
|
|
// Clear buffered_messages
|
2017-08-14 00:44:45 +02:00
|
|
|
for (auto &buffered_message : buffered_messages) {
|
|
|
|
delete buffered_message.second;
|
2010-11-27 00:02:21 +01:00
|
|
|
}
|
2010-12-18 12:10:37 +01:00
|
|
|
}
|
2010-11-27 00:02:21 +01:00
|
|
|
|
2011-02-23 01:49:57 +01:00
|
|
|
/*
|
|
|
|
Send queued-for-sending map edit events.
|
|
|
|
*/
|
|
|
|
{
|
2014-01-31 00:24:00 +01:00
|
|
|
// We will be accessing the environment
|
2015-04-07 12:13:12 +02:00
|
|
|
MutexAutoLock lock(m_env_mutex);
|
2012-03-28 11:51:47 +02:00
|
|
|
|
2011-06-25 03:25:14 +02:00
|
|
|
// Don't send too many at a time
|
2011-06-26 02:14:52 +02:00
|
|
|
//u32 count = 0;
|
|
|
|
|
|
|
|
// Single change sending is disabled if queue size is not small
|
|
|
|
bool disable_single_change_sending = false;
|
|
|
|
if(m_unsent_map_edit_queue.size() >= 4)
|
|
|
|
disable_single_change_sending = true;
|
|
|
|
|
2012-03-11 03:15:45 +01:00
|
|
|
int event_count = m_unsent_map_edit_queue.size();
|
2011-06-26 23:27:17 +02:00
|
|
|
|
|
|
|
// We'll log the amount of each
|
|
|
|
Profiler prof;
|
|
|
|
|
2018-12-04 20:37:48 +01:00
|
|
|
std::list<v3s16> node_meta_updates;
|
|
|
|
|
2017-08-14 00:44:45 +02:00
|
|
|
while (!m_unsent_map_edit_queue.empty()) {
|
2015-03-04 17:48:07 +01:00
|
|
|
MapEditEvent* event = m_unsent_map_edit_queue.front();
|
|
|
|
m_unsent_map_edit_queue.pop();
|
2012-11-26 03:16:48 +01:00
|
|
|
|
2011-06-25 03:25:14 +02:00
|
|
|
// Players far away from the change are stored here.
|
|
|
|
// Instead of sending the changes, MapBlocks are set not sent
|
|
|
|
// for them.
|
2018-08-16 20:10:34 +02:00
|
|
|
std::unordered_set<u16> far_players;
|
2011-02-23 01:49:57 +01:00
|
|
|
|
2015-03-05 11:18:17 +01:00
|
|
|
switch (event->type) {
|
|
|
|
case MEET_ADDNODE:
|
|
|
|
case MEET_SWAPNODE:
|
2011-06-26 23:27:17 +02:00
|
|
|
prof.add("MEET_ADDNODE", 1);
|
2018-08-16 20:10:34 +02:00
|
|
|
sendAddNode(event->p, event->n, &far_players,
|
|
|
|
disable_single_change_sending ? 5 : 30,
|
2015-03-05 11:18:17 +01:00
|
|
|
event->type == MEET_ADDNODE);
|
|
|
|
break;
|
|
|
|
case MEET_REMOVENODE:
|
2011-06-26 23:27:17 +02:00
|
|
|
prof.add("MEET_REMOVENODE", 1);
|
2018-08-16 20:10:34 +02:00
|
|
|
sendRemoveNode(event->p, &far_players,
|
|
|
|
disable_single_change_sending ? 5 : 30);
|
2015-03-05 11:18:17 +01:00
|
|
|
break;
|
2018-12-04 20:37:48 +01:00
|
|
|
case MEET_BLOCK_NODE_METADATA_CHANGED: {
|
|
|
|
verbosestream << "Server: MEET_BLOCK_NODE_METADATA_CHANGED" << std::endl;
|
|
|
|
prof.add("MEET_BLOCK_NODE_METADATA_CHANGED", 1);
|
|
|
|
if (!event->is_private_change) {
|
|
|
|
// Don't send the change yet. Collect them to eliminate dupes.
|
|
|
|
node_meta_updates.remove(event->p);
|
|
|
|
node_meta_updates.push_back(event->p);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (MapBlock *block = m_env->getMap().getBlockNoCreateNoEx(
|
|
|
|
getNodeBlockPos(event->p))) {
|
|
|
|
block->raiseModified(MOD_STATE_WRITE_NEEDED,
|
|
|
|
MOD_REASON_REPORT_META_CHANGE);
|
|
|
|
}
|
2015-03-05 11:18:17 +01:00
|
|
|
break;
|
2018-12-04 20:37:48 +01:00
|
|
|
}
|
2015-03-05 11:18:17 +01:00
|
|
|
case MEET_OTHER:
|
|
|
|
infostream << "Server: MEET_OTHER" << std::endl;
|
2011-06-26 23:27:17 +02:00
|
|
|
prof.add("MEET_OTHER", 1);
|
2017-08-14 00:44:45 +02:00
|
|
|
for (const v3s16 &modified_block : event->modified_blocks) {
|
2018-03-08 22:58:43 +01:00
|
|
|
m_clients.markBlockposAsNotSent(modified_block);
|
2011-08-16 11:23:19 +02:00
|
|
|
}
|
2015-03-05 11:18:17 +01:00
|
|
|
break;
|
|
|
|
default:
|
2011-06-26 23:27:17 +02:00
|
|
|
prof.add("unknown", 1);
|
2015-10-14 07:26:03 +02:00
|
|
|
warningstream << "Server: Unknown MapEditEvent "
|
2015-03-05 11:18:17 +01:00
|
|
|
<< ((u32)event->type) << std::endl;
|
|
|
|
break;
|
2011-02-23 01:49:57 +01:00
|
|
|
}
|
2012-11-26 03:16:48 +01:00
|
|
|
|
2011-06-25 03:25:14 +02:00
|
|
|
/*
|
|
|
|
Set blocks not sent to far players
|
|
|
|
*/
|
2017-08-14 00:44:45 +02:00
|
|
|
if (!far_players.empty()) {
|
2011-07-01 20:04:40 +02:00
|
|
|
// Convert list format to that wanted by SetBlocksNotSent
|
2012-12-20 18:19:49 +01:00
|
|
|
std::map<v3s16, MapBlock*> modified_blocks2;
|
2017-08-14 00:44:45 +02:00
|
|
|
for (const v3s16 &modified_block : event->modified_blocks) {
|
|
|
|
modified_blocks2[modified_block] =
|
|
|
|
m_env->getMap().getBlockNoCreateNoEx(modified_block);
|
2011-06-26 02:14:52 +02:00
|
|
|
}
|
2015-03-05 11:18:17 +01:00
|
|
|
|
2011-07-01 20:04:40 +02:00
|
|
|
// Set blocks not sent
|
2017-08-14 00:44:45 +02:00
|
|
|
for (const u16 far_player : far_players) {
|
|
|
|
if (RemoteClient *client = getClient(far_player))
|
2015-03-05 11:18:17 +01:00
|
|
|
client->SetBlocksNotSent(modified_blocks2);
|
2011-06-26 02:14:52 +02:00
|
|
|
}
|
2011-06-25 03:25:14 +02:00
|
|
|
}
|
2011-02-23 01:49:57 +01:00
|
|
|
|
|
|
|
delete event;
|
|
|
|
}
|
2011-06-26 23:27:17 +02:00
|
|
|
|
2017-08-14 00:44:45 +02:00
|
|
|
if (event_count >= 5) {
|
|
|
|
infostream << "Server: MapEditEvents:" << std::endl;
|
2011-10-16 13:57:53 +02:00
|
|
|
prof.print(infostream);
|
2017-08-14 00:44:45 +02:00
|
|
|
} else if (event_count != 0) {
|
|
|
|
verbosestream << "Server: MapEditEvents:" << std::endl;
|
2012-03-11 03:15:45 +01:00
|
|
|
prof.print(verbosestream);
|
2011-06-26 23:27:17 +02:00
|
|
|
}
|
2012-11-26 03:16:48 +01:00
|
|
|
|
2018-12-04 20:37:48 +01:00
|
|
|
// Send all metadata updates
|
|
|
|
if (node_meta_updates.size())
|
|
|
|
sendMetadataChanged(node_meta_updates);
|
2011-02-23 01:49:57 +01:00
|
|
|
}
|
|
|
|
|
2011-02-05 13:55:16 +01:00
|
|
|
/*
|
|
|
|
Trigger emergethread (it somehow gets to a non-triggered but
|
|
|
|
bysy state sometimes)
|
|
|
|
*/
|
2010-11-29 09:52:07 +01:00
|
|
|
{
|
2010-12-19 15:51:45 +01:00
|
|
|
float &counter = m_emergethread_trigger_timer;
|
2010-11-29 09:52:07 +01:00
|
|
|
counter += dtime;
|
2015-09-02 21:21:55 +02:00
|
|
|
if (counter >= 2.0) {
|
2010-11-29 09:52:07 +01:00
|
|
|
counter = 0.0;
|
2012-11-26 03:16:48 +01:00
|
|
|
|
2014-01-26 07:12:18 +01:00
|
|
|
m_emerge->startThreads();
|
2010-11-29 09:52:07 +01:00
|
|
|
}
|
|
|
|
}
|
2010-11-27 00:02:21 +01:00
|
|
|
|
2011-05-29 20:11:16 +02:00
|
|
|
// Save map, players and auth stuff
|
2010-11-27 00:02:21 +01:00
|
|
|
{
|
2010-12-19 15:51:45 +01:00
|
|
|
float &counter = m_savemap_timer;
|
2010-11-27 00:02:21 +01:00
|
|
|
counter += dtime;
|
2017-06-11 13:58:43 +02:00
|
|
|
static thread_local const float save_interval =
|
2015-09-02 21:21:55 +02:00
|
|
|
g_settings->getFloat("server_map_save_interval");
|
|
|
|
if (counter >= save_interval) {
|
2010-11-27 00:02:21 +01:00
|
|
|
counter = 0.0;
|
2015-04-07 12:13:12 +02:00
|
|
|
MutexAutoLock lock(m_env_mutex);
|
2010-11-27 00:02:21 +01:00
|
|
|
|
2011-10-12 12:53:38 +02:00
|
|
|
ScopeProfiler sp(g_profiler, "Server: saving stuff");
|
2011-05-30 23:15:43 +02:00
|
|
|
|
2014-05-30 22:04:07 +02:00
|
|
|
// Save ban file
|
|
|
|
if (m_banmanager->isModified()) {
|
2013-08-11 04:09:45 +02:00
|
|
|
m_banmanager->save();
|
2014-05-30 22:04:07 +02:00
|
|
|
}
|
2012-11-26 03:16:48 +01:00
|
|
|
|
2011-11-27 23:45:34 +01:00
|
|
|
// Save changed parts of map
|
|
|
|
m_env->getMap().save(MOD_STATE_WRITE_NEEDED);
|
2011-06-26 23:27:17 +02:00
|
|
|
|
|
|
|
// Save players
|
2014-05-30 22:04:07 +02:00
|
|
|
m_env->saveLoadedPlayers();
|
2012-11-26 03:16:48 +01:00
|
|
|
|
2011-06-26 23:27:17 +02:00
|
|
|
// Save environment metadata
|
2014-05-30 22:04:07 +02:00
|
|
|
m_env->saveMeta();
|
2010-11-27 00:02:21 +01:00
|
|
|
}
|
|
|
|
}
|
2017-04-15 23:19:18 +02:00
|
|
|
|
2018-06-13 21:58:34 +02:00
|
|
|
m_shutdown_state.tick(dtime, this);
|
2010-11-27 00:02:21 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void Server::Receive()
|
|
|
|
{
|
2018-12-04 12:39:10 +01:00
|
|
|
session_t peer_id = 0;
|
2015-01-12 17:01:41 +01:00
|
|
|
try {
|
2015-03-31 10:35:51 +02:00
|
|
|
NetworkPacket pkt;
|
2017-08-24 08:28:54 +02:00
|
|
|
m_con->Receive(&pkt);
|
2015-03-31 10:35:51 +02:00
|
|
|
peer_id = pkt.getPeerId();
|
|
|
|
ProcessData(&pkt);
|
2017-08-18 12:17:30 +02:00
|
|
|
} catch (const con::InvalidIncomingDataException &e) {
|
|
|
|
infostream << "Server::Receive(): InvalidIncomingDataException: what()="
|
|
|
|
<< e.what() << std::endl;
|
|
|
|
} catch (const SerializationError &e) {
|
|
|
|
infostream << "Server::Receive(): SerializationError: what()="
|
|
|
|
<< e.what() << std::endl;
|
|
|
|
} catch (const ClientStateError &e) {
|
2017-08-14 01:06:12 +02:00
|
|
|
errorstream << "ProcessData: peer=" << peer_id << e.what() << std::endl;
|
2015-03-13 16:35:21 +01:00
|
|
|
DenyAccess_Legacy(peer_id, L"Your client sent something server didn't expect."
|
2014-02-13 20:17:42 +01:00
|
|
|
L"Try reconnecting or updating your client");
|
2017-08-18 12:17:30 +02:00
|
|
|
} catch (const con::PeerNotFoundException &e) {
|
[Patch 2/4] Network rework: packet writing, sending and cleanups
NetworkPacket.cpp:
* Remove some deprecated functions, we must use streaming interface
* m_data converted from u8* to std::vector<u8>
* Add an exporter to forge packet to Connection object
* implement operator << std::wstring. n
* implement operator << std::string
* dynamic resize when write packet content.
* fix string writing and performances.
* create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Reliability
* Transmit channel
* Implement putRawString for some ugly char (_INIT packet), and use it.
* Many packet read and write migrated
* Implement oldForgePacket to interface writing with current connection
* fix U8/char/bool writing
* fix string writing and performances.
* add some missing functions
* Use v3s16 read instead of reading x,y,z separately
* Add irr::video::SColor support into packets
* Add some missing handlers
* Add a template function to increase offset
* Throw a serialization error on packet reading (must be improved)
PacketFactories:
* Create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Create ClientCommandFactory, used by server to get useful informations about packet processing (sending).
Client.cpp:
* implement NetworkPacket ::Send interface.
* Move packet handlers to a dedicated file
* Remove Client::Send(SharedBuffer)
Server.cpp:
* implement NetworkPacket ::Send interface.
* Rewrite all packets using NetworkPacket
* Move packet handlers to a dedicated file
* Remove Server::Send(SharedBuffer)
ClientIface.cpp:
* Remove sendToAll(SharedBuffer<u8>)
Connection.hpp rework:
* Remove duplicate include
* Remove duplicate negation
* Remove a useless variable
* Improve code performance by using a m_peers_list instead of scanning m_peers map
* Remove Connection::Send(SharedBuffer)
* Fix useafterfree into NetworkPacket Sending
* Remove unused Connection::sendToAll
Test.cpp:
* Remove dead code
* Update tests to use NetworkPackets
Misc:
* add new wrappers to Send packets in client, using NetworkPacket
* Add NetworkPacket methods for Connection
* coding style fix
* dead code since changes cleanup
* Use v3s16 read instead of reading x,y,z separately in some packets
* Use different files to handle packets received by client and server
* Cleanup: Remove useless includes
ok @Zeno-
Tested by @Zeno- @VanessaE and @nerzhul on running servers
2015-01-16 11:37:49 +01:00
|
|
|
// Do nothing
|
2015-01-12 17:01:41 +01:00
|
|
|
}
|
|
|
|
}
|
2011-11-29 16:15:18 +01:00
|
|
|
|
2017-09-27 19:47:36 +02:00
|
|
|
PlayerSAO* Server::StageTwoClientInit(session_t peer_id)
|
2015-01-12 17:01:41 +01:00
|
|
|
{
|
2017-08-14 01:06:12 +02:00
|
|
|
std::string playername;
|
[Patch 2/4] Network rework: packet writing, sending and cleanups
NetworkPacket.cpp:
* Remove some deprecated functions, we must use streaming interface
* m_data converted from u8* to std::vector<u8>
* Add an exporter to forge packet to Connection object
* implement operator << std::wstring. n
* implement operator << std::string
* dynamic resize when write packet content.
* fix string writing and performances.
* create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Reliability
* Transmit channel
* Implement putRawString for some ugly char (_INIT packet), and use it.
* Many packet read and write migrated
* Implement oldForgePacket to interface writing with current connection
* fix U8/char/bool writing
* fix string writing and performances.
* add some missing functions
* Use v3s16 read instead of reading x,y,z separately
* Add irr::video::SColor support into packets
* Add some missing handlers
* Add a template function to increase offset
* Throw a serialization error on packet reading (must be improved)
PacketFactories:
* Create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Create ClientCommandFactory, used by server to get useful informations about packet processing (sending).
Client.cpp:
* implement NetworkPacket ::Send interface.
* Move packet handlers to a dedicated file
* Remove Client::Send(SharedBuffer)
Server.cpp:
* implement NetworkPacket ::Send interface.
* Rewrite all packets using NetworkPacket
* Move packet handlers to a dedicated file
* Remove Server::Send(SharedBuffer)
ClientIface.cpp:
* Remove sendToAll(SharedBuffer<u8>)
Connection.hpp rework:
* Remove duplicate include
* Remove duplicate negation
* Remove a useless variable
* Improve code performance by using a m_peers_list instead of scanning m_peers map
* Remove Connection::Send(SharedBuffer)
* Fix useafterfree into NetworkPacket Sending
* Remove unused Connection::sendToAll
Test.cpp:
* Remove dead code
* Update tests to use NetworkPackets
Misc:
* add new wrappers to Send packets in client, using NetworkPacket
* Add NetworkPacket methods for Connection
* coding style fix
* dead code since changes cleanup
* Use v3s16 read instead of reading x,y,z separately in some packets
* Use different files to handle packets received by client and server
* Cleanup: Remove useless includes
ok @Zeno-
Tested by @Zeno- @VanessaE and @nerzhul on running servers
2015-01-16 11:37:49 +01:00
|
|
|
PlayerSAO *playersao = NULL;
|
2015-04-07 12:13:12 +02:00
|
|
|
m_clients.lock();
|
[Patch 2/4] Network rework: packet writing, sending and cleanups
NetworkPacket.cpp:
* Remove some deprecated functions, we must use streaming interface
* m_data converted from u8* to std::vector<u8>
* Add an exporter to forge packet to Connection object
* implement operator << std::wstring. n
* implement operator << std::string
* dynamic resize when write packet content.
* fix string writing and performances.
* create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Reliability
* Transmit channel
* Implement putRawString for some ugly char (_INIT packet), and use it.
* Many packet read and write migrated
* Implement oldForgePacket to interface writing with current connection
* fix U8/char/bool writing
* fix string writing and performances.
* add some missing functions
* Use v3s16 read instead of reading x,y,z separately
* Add irr::video::SColor support into packets
* Add some missing handlers
* Add a template function to increase offset
* Throw a serialization error on packet reading (must be improved)
PacketFactories:
* Create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Create ClientCommandFactory, used by server to get useful informations about packet processing (sending).
Client.cpp:
* implement NetworkPacket ::Send interface.
* Move packet handlers to a dedicated file
* Remove Client::Send(SharedBuffer)
Server.cpp:
* implement NetworkPacket ::Send interface.
* Rewrite all packets using NetworkPacket
* Move packet handlers to a dedicated file
* Remove Server::Send(SharedBuffer)
ClientIface.cpp:
* Remove sendToAll(SharedBuffer<u8>)
Connection.hpp rework:
* Remove duplicate include
* Remove duplicate negation
* Remove a useless variable
* Improve code performance by using a m_peers_list instead of scanning m_peers map
* Remove Connection::Send(SharedBuffer)
* Fix useafterfree into NetworkPacket Sending
* Remove unused Connection::sendToAll
Test.cpp:
* Remove dead code
* Update tests to use NetworkPackets
Misc:
* add new wrappers to Send packets in client, using NetworkPacket
* Add NetworkPacket methods for Connection
* coding style fix
* dead code since changes cleanup
* Use v3s16 read instead of reading x,y,z separately in some packets
* Use different files to handle packets received by client and server
* Cleanup: Remove useless includes
ok @Zeno-
Tested by @Zeno- @VanessaE and @nerzhul on running servers
2015-01-16 11:37:49 +01:00
|
|
|
try {
|
|
|
|
RemoteClient* client = m_clients.lockedGetClientNoEx(peer_id, CS_InitDone);
|
2017-08-15 09:30:31 +02:00
|
|
|
if (client) {
|
[Patch 2/4] Network rework: packet writing, sending and cleanups
NetworkPacket.cpp:
* Remove some deprecated functions, we must use streaming interface
* m_data converted from u8* to std::vector<u8>
* Add an exporter to forge packet to Connection object
* implement operator << std::wstring. n
* implement operator << std::string
* dynamic resize when write packet content.
* fix string writing and performances.
* create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Reliability
* Transmit channel
* Implement putRawString for some ugly char (_INIT packet), and use it.
* Many packet read and write migrated
* Implement oldForgePacket to interface writing with current connection
* fix U8/char/bool writing
* fix string writing and performances.
* add some missing functions
* Use v3s16 read instead of reading x,y,z separately
* Add irr::video::SColor support into packets
* Add some missing handlers
* Add a template function to increase offset
* Throw a serialization error on packet reading (must be improved)
PacketFactories:
* Create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Create ClientCommandFactory, used by server to get useful informations about packet processing (sending).
Client.cpp:
* implement NetworkPacket ::Send interface.
* Move packet handlers to a dedicated file
* Remove Client::Send(SharedBuffer)
Server.cpp:
* implement NetworkPacket ::Send interface.
* Rewrite all packets using NetworkPacket
* Move packet handlers to a dedicated file
* Remove Server::Send(SharedBuffer)
ClientIface.cpp:
* Remove sendToAll(SharedBuffer<u8>)
Connection.hpp rework:
* Remove duplicate include
* Remove duplicate negation
* Remove a useless variable
* Improve code performance by using a m_peers_list instead of scanning m_peers map
* Remove Connection::Send(SharedBuffer)
* Fix useafterfree into NetworkPacket Sending
* Remove unused Connection::sendToAll
Test.cpp:
* Remove dead code
* Update tests to use NetworkPackets
Misc:
* add new wrappers to Send packets in client, using NetworkPacket
* Add NetworkPacket methods for Connection
* coding style fix
* dead code since changes cleanup
* Use v3s16 read instead of reading x,y,z separately in some packets
* Use different files to handle packets received by client and server
* Cleanup: Remove useless includes
ok @Zeno-
Tested by @Zeno- @VanessaE and @nerzhul on running servers
2015-01-16 11:37:49 +01:00
|
|
|
playername = client->getName();
|
2015-07-17 16:40:41 +02:00
|
|
|
playersao = emergePlayer(playername.c_str(), peer_id, client->net_proto_version);
|
[Patch 2/4] Network rework: packet writing, sending and cleanups
NetworkPacket.cpp:
* Remove some deprecated functions, we must use streaming interface
* m_data converted from u8* to std::vector<u8>
* Add an exporter to forge packet to Connection object
* implement operator << std::wstring. n
* implement operator << std::string
* dynamic resize when write packet content.
* fix string writing and performances.
* create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Reliability
* Transmit channel
* Implement putRawString for some ugly char (_INIT packet), and use it.
* Many packet read and write migrated
* Implement oldForgePacket to interface writing with current connection
* fix U8/char/bool writing
* fix string writing and performances.
* add some missing functions
* Use v3s16 read instead of reading x,y,z separately
* Add irr::video::SColor support into packets
* Add some missing handlers
* Add a template function to increase offset
* Throw a serialization error on packet reading (must be improved)
PacketFactories:
* Create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Create ClientCommandFactory, used by server to get useful informations about packet processing (sending).
Client.cpp:
* implement NetworkPacket ::Send interface.
* Move packet handlers to a dedicated file
* Remove Client::Send(SharedBuffer)
Server.cpp:
* implement NetworkPacket ::Send interface.
* Rewrite all packets using NetworkPacket
* Move packet handlers to a dedicated file
* Remove Server::Send(SharedBuffer)
ClientIface.cpp:
* Remove sendToAll(SharedBuffer<u8>)
Connection.hpp rework:
* Remove duplicate include
* Remove duplicate negation
* Remove a useless variable
* Improve code performance by using a m_peers_list instead of scanning m_peers map
* Remove Connection::Send(SharedBuffer)
* Fix useafterfree into NetworkPacket Sending
* Remove unused Connection::sendToAll
Test.cpp:
* Remove dead code
* Update tests to use NetworkPackets
Misc:
* add new wrappers to Send packets in client, using NetworkPacket
* Add NetworkPacket methods for Connection
* coding style fix
* dead code since changes cleanup
* Use v3s16 read instead of reading x,y,z separately in some packets
* Use different files to handle packets received by client and server
* Cleanup: Remove useless includes
ok @Zeno-
Tested by @Zeno- @VanessaE and @nerzhul on running servers
2015-01-16 11:37:49 +01:00
|
|
|
}
|
|
|
|
} catch (std::exception &e) {
|
2015-04-07 12:13:12 +02:00
|
|
|
m_clients.unlock();
|
[Patch 2/4] Network rework: packet writing, sending and cleanups
NetworkPacket.cpp:
* Remove some deprecated functions, we must use streaming interface
* m_data converted from u8* to std::vector<u8>
* Add an exporter to forge packet to Connection object
* implement operator << std::wstring. n
* implement operator << std::string
* dynamic resize when write packet content.
* fix string writing and performances.
* create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Reliability
* Transmit channel
* Implement putRawString for some ugly char (_INIT packet), and use it.
* Many packet read and write migrated
* Implement oldForgePacket to interface writing with current connection
* fix U8/char/bool writing
* fix string writing and performances.
* add some missing functions
* Use v3s16 read instead of reading x,y,z separately
* Add irr::video::SColor support into packets
* Add some missing handlers
* Add a template function to increase offset
* Throw a serialization error on packet reading (must be improved)
PacketFactories:
* Create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Create ClientCommandFactory, used by server to get useful informations about packet processing (sending).
Client.cpp:
* implement NetworkPacket ::Send interface.
* Move packet handlers to a dedicated file
* Remove Client::Send(SharedBuffer)
Server.cpp:
* implement NetworkPacket ::Send interface.
* Rewrite all packets using NetworkPacket
* Move packet handlers to a dedicated file
* Remove Server::Send(SharedBuffer)
ClientIface.cpp:
* Remove sendToAll(SharedBuffer<u8>)
Connection.hpp rework:
* Remove duplicate include
* Remove duplicate negation
* Remove a useless variable
* Improve code performance by using a m_peers_list instead of scanning m_peers map
* Remove Connection::Send(SharedBuffer)
* Fix useafterfree into NetworkPacket Sending
* Remove unused Connection::sendToAll
Test.cpp:
* Remove dead code
* Update tests to use NetworkPackets
Misc:
* add new wrappers to Send packets in client, using NetworkPacket
* Add NetworkPacket methods for Connection
* coding style fix
* dead code since changes cleanup
* Use v3s16 read instead of reading x,y,z separately in some packets
* Use different files to handle packets received by client and server
* Cleanup: Remove useless includes
ok @Zeno-
Tested by @Zeno- @VanessaE and @nerzhul on running servers
2015-01-16 11:37:49 +01:00
|
|
|
throw;
|
|
|
|
}
|
2015-04-07 12:13:12 +02:00
|
|
|
m_clients.unlock();
|
2012-01-21 00:11:44 +01:00
|
|
|
|
2017-01-01 16:13:01 +01:00
|
|
|
RemotePlayer *player = m_env->getPlayer(playername.c_str());
|
2011-11-29 16:15:18 +01:00
|
|
|
|
[Patch 2/4] Network rework: packet writing, sending and cleanups
NetworkPacket.cpp:
* Remove some deprecated functions, we must use streaming interface
* m_data converted from u8* to std::vector<u8>
* Add an exporter to forge packet to Connection object
* implement operator << std::wstring. n
* implement operator << std::string
* dynamic resize when write packet content.
* fix string writing and performances.
* create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Reliability
* Transmit channel
* Implement putRawString for some ugly char (_INIT packet), and use it.
* Many packet read and write migrated
* Implement oldForgePacket to interface writing with current connection
* fix U8/char/bool writing
* fix string writing and performances.
* add some missing functions
* Use v3s16 read instead of reading x,y,z separately
* Add irr::video::SColor support into packets
* Add some missing handlers
* Add a template function to increase offset
* Throw a serialization error on packet reading (must be improved)
PacketFactories:
* Create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Create ClientCommandFactory, used by server to get useful informations about packet processing (sending).
Client.cpp:
* implement NetworkPacket ::Send interface.
* Move packet handlers to a dedicated file
* Remove Client::Send(SharedBuffer)
Server.cpp:
* implement NetworkPacket ::Send interface.
* Rewrite all packets using NetworkPacket
* Move packet handlers to a dedicated file
* Remove Server::Send(SharedBuffer)
ClientIface.cpp:
* Remove sendToAll(SharedBuffer<u8>)
Connection.hpp rework:
* Remove duplicate include
* Remove duplicate negation
* Remove a useless variable
* Improve code performance by using a m_peers_list instead of scanning m_peers map
* Remove Connection::Send(SharedBuffer)
* Fix useafterfree into NetworkPacket Sending
* Remove unused Connection::sendToAll
Test.cpp:
* Remove dead code
* Update tests to use NetworkPackets
Misc:
* add new wrappers to Send packets in client, using NetworkPacket
* Add NetworkPacket methods for Connection
* coding style fix
* dead code since changes cleanup
* Use v3s16 read instead of reading x,y,z separately in some packets
* Use different files to handle packets received by client and server
* Cleanup: Remove useless includes
ok @Zeno-
Tested by @Zeno- @VanessaE and @nerzhul on running servers
2015-01-16 11:37:49 +01:00
|
|
|
// If failed, cancel
|
2017-08-15 09:30:31 +02:00
|
|
|
if (!playersao || !player) {
|
2017-09-30 12:00:05 +02:00
|
|
|
if (player && player->getPeerId() != PEER_ID_INEXISTENT) {
|
2015-05-19 00:55:58 +02:00
|
|
|
actionstream << "Server: Failed to emerge player \"" << playername
|
|
|
|
<< "\" (player allocated to an another client)" << std::endl;
|
2015-03-13 16:35:21 +01:00
|
|
|
DenyAccess_Legacy(peer_id, L"Another client is connected with this "
|
[Patch 2/4] Network rework: packet writing, sending and cleanups
NetworkPacket.cpp:
* Remove some deprecated functions, we must use streaming interface
* m_data converted from u8* to std::vector<u8>
* Add an exporter to forge packet to Connection object
* implement operator << std::wstring. n
* implement operator << std::string
* dynamic resize when write packet content.
* fix string writing and performances.
* create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Reliability
* Transmit channel
* Implement putRawString for some ugly char (_INIT packet), and use it.
* Many packet read and write migrated
* Implement oldForgePacket to interface writing with current connection
* fix U8/char/bool writing
* fix string writing and performances.
* add some missing functions
* Use v3s16 read instead of reading x,y,z separately
* Add irr::video::SColor support into packets
* Add some missing handlers
* Add a template function to increase offset
* Throw a serialization error on packet reading (must be improved)
PacketFactories:
* Create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Create ClientCommandFactory, used by server to get useful informations about packet processing (sending).
Client.cpp:
* implement NetworkPacket ::Send interface.
* Move packet handlers to a dedicated file
* Remove Client::Send(SharedBuffer)
Server.cpp:
* implement NetworkPacket ::Send interface.
* Rewrite all packets using NetworkPacket
* Move packet handlers to a dedicated file
* Remove Server::Send(SharedBuffer)
ClientIface.cpp:
* Remove sendToAll(SharedBuffer<u8>)
Connection.hpp rework:
* Remove duplicate include
* Remove duplicate negation
* Remove a useless variable
* Improve code performance by using a m_peers_list instead of scanning m_peers map
* Remove Connection::Send(SharedBuffer)
* Fix useafterfree into NetworkPacket Sending
* Remove unused Connection::sendToAll
Test.cpp:
* Remove dead code
* Update tests to use NetworkPackets
Misc:
* add new wrappers to Send packets in client, using NetworkPacket
* Add NetworkPacket methods for Connection
* coding style fix
* dead code since changes cleanup
* Use v3s16 read instead of reading x,y,z separately in some packets
* Use different files to handle packets received by client and server
* Cleanup: Remove useless includes
ok @Zeno-
Tested by @Zeno- @VanessaE and @nerzhul on running servers
2015-01-16 11:37:49 +01:00
|
|
|
L"name. If your client closed unexpectedly, try again in "
|
|
|
|
L"a minute.");
|
|
|
|
} else {
|
2015-05-19 00:55:58 +02:00
|
|
|
errorstream << "Server: " << playername << ": Failed to emerge player"
|
|
|
|
<< std::endl;
|
2015-03-13 16:35:21 +01:00
|
|
|
DenyAccess_Legacy(peer_id, L"Could not allocate player.");
|
[Patch 2/4] Network rework: packet writing, sending and cleanups
NetworkPacket.cpp:
* Remove some deprecated functions, we must use streaming interface
* m_data converted from u8* to std::vector<u8>
* Add an exporter to forge packet to Connection object
* implement operator << std::wstring. n
* implement operator << std::string
* dynamic resize when write packet content.
* fix string writing and performances.
* create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Reliability
* Transmit channel
* Implement putRawString for some ugly char (_INIT packet), and use it.
* Many packet read and write migrated
* Implement oldForgePacket to interface writing with current connection
* fix U8/char/bool writing
* fix string writing and performances.
* add some missing functions
* Use v3s16 read instead of reading x,y,z separately
* Add irr::video::SColor support into packets
* Add some missing handlers
* Add a template function to increase offset
* Throw a serialization error on packet reading (must be improved)
PacketFactories:
* Create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Create ClientCommandFactory, used by server to get useful informations about packet processing (sending).
Client.cpp:
* implement NetworkPacket ::Send interface.
* Move packet handlers to a dedicated file
* Remove Client::Send(SharedBuffer)
Server.cpp:
* implement NetworkPacket ::Send interface.
* Rewrite all packets using NetworkPacket
* Move packet handlers to a dedicated file
* Remove Server::Send(SharedBuffer)
ClientIface.cpp:
* Remove sendToAll(SharedBuffer<u8>)
Connection.hpp rework:
* Remove duplicate include
* Remove duplicate negation
* Remove a useless variable
* Improve code performance by using a m_peers_list instead of scanning m_peers map
* Remove Connection::Send(SharedBuffer)
* Fix useafterfree into NetworkPacket Sending
* Remove unused Connection::sendToAll
Test.cpp:
* Remove dead code
* Update tests to use NetworkPackets
Misc:
* add new wrappers to Send packets in client, using NetworkPacket
* Add NetworkPacket methods for Connection
* coding style fix
* dead code since changes cleanup
* Use v3s16 read instead of reading x,y,z separately in some packets
* Use different files to handle packets received by client and server
* Cleanup: Remove useless includes
ok @Zeno-
Tested by @Zeno- @VanessaE and @nerzhul on running servers
2015-01-16 11:37:49 +01:00
|
|
|
}
|
|
|
|
return NULL;
|
2015-01-12 17:01:41 +01:00
|
|
|
}
|
2011-11-29 16:15:18 +01:00
|
|
|
|
[Patch 2/4] Network rework: packet writing, sending and cleanups
NetworkPacket.cpp:
* Remove some deprecated functions, we must use streaming interface
* m_data converted from u8* to std::vector<u8>
* Add an exporter to forge packet to Connection object
* implement operator << std::wstring. n
* implement operator << std::string
* dynamic resize when write packet content.
* fix string writing and performances.
* create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Reliability
* Transmit channel
* Implement putRawString for some ugly char (_INIT packet), and use it.
* Many packet read and write migrated
* Implement oldForgePacket to interface writing with current connection
* fix U8/char/bool writing
* fix string writing and performances.
* add some missing functions
* Use v3s16 read instead of reading x,y,z separately
* Add irr::video::SColor support into packets
* Add some missing handlers
* Add a template function to increase offset
* Throw a serialization error on packet reading (must be improved)
PacketFactories:
* Create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Create ClientCommandFactory, used by server to get useful informations about packet processing (sending).
Client.cpp:
* implement NetworkPacket ::Send interface.
* Move packet handlers to a dedicated file
* Remove Client::Send(SharedBuffer)
Server.cpp:
* implement NetworkPacket ::Send interface.
* Rewrite all packets using NetworkPacket
* Move packet handlers to a dedicated file
* Remove Server::Send(SharedBuffer)
ClientIface.cpp:
* Remove sendToAll(SharedBuffer<u8>)
Connection.hpp rework:
* Remove duplicate include
* Remove duplicate negation
* Remove a useless variable
* Improve code performance by using a m_peers_list instead of scanning m_peers map
* Remove Connection::Send(SharedBuffer)
* Fix useafterfree into NetworkPacket Sending
* Remove unused Connection::sendToAll
Test.cpp:
* Remove dead code
* Update tests to use NetworkPackets
Misc:
* add new wrappers to Send packets in client, using NetworkPacket
* Add NetworkPacket methods for Connection
* coding style fix
* dead code since changes cleanup
* Use v3s16 read instead of reading x,y,z separately in some packets
* Use different files to handle packets received by client and server
* Cleanup: Remove useless includes
ok @Zeno-
Tested by @Zeno- @VanessaE and @nerzhul on running servers
2015-01-16 11:37:49 +01:00
|
|
|
/*
|
|
|
|
Send complete position information
|
|
|
|
*/
|
|
|
|
SendMovePlayer(peer_id);
|
2011-11-29 16:15:18 +01:00
|
|
|
|
[Patch 2/4] Network rework: packet writing, sending and cleanups
NetworkPacket.cpp:
* Remove some deprecated functions, we must use streaming interface
* m_data converted from u8* to std::vector<u8>
* Add an exporter to forge packet to Connection object
* implement operator << std::wstring. n
* implement operator << std::string
* dynamic resize when write packet content.
* fix string writing and performances.
* create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Reliability
* Transmit channel
* Implement putRawString for some ugly char (_INIT packet), and use it.
* Many packet read and write migrated
* Implement oldForgePacket to interface writing with current connection
* fix U8/char/bool writing
* fix string writing and performances.
* add some missing functions
* Use v3s16 read instead of reading x,y,z separately
* Add irr::video::SColor support into packets
* Add some missing handlers
* Add a template function to increase offset
* Throw a serialization error on packet reading (must be improved)
PacketFactories:
* Create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Create ClientCommandFactory, used by server to get useful informations about packet processing (sending).
Client.cpp:
* implement NetworkPacket ::Send interface.
* Move packet handlers to a dedicated file
* Remove Client::Send(SharedBuffer)
Server.cpp:
* implement NetworkPacket ::Send interface.
* Rewrite all packets using NetworkPacket
* Move packet handlers to a dedicated file
* Remove Server::Send(SharedBuffer)
ClientIface.cpp:
* Remove sendToAll(SharedBuffer<u8>)
Connection.hpp rework:
* Remove duplicate include
* Remove duplicate negation
* Remove a useless variable
* Improve code performance by using a m_peers_list instead of scanning m_peers map
* Remove Connection::Send(SharedBuffer)
* Fix useafterfree into NetworkPacket Sending
* Remove unused Connection::sendToAll
Test.cpp:
* Remove dead code
* Update tests to use NetworkPackets
Misc:
* add new wrappers to Send packets in client, using NetworkPacket
* Add NetworkPacket methods for Connection
* coding style fix
* dead code since changes cleanup
* Use v3s16 read instead of reading x,y,z separately in some packets
* Use different files to handle packets received by client and server
* Cleanup: Remove useless includes
ok @Zeno-
Tested by @Zeno- @VanessaE and @nerzhul on running servers
2015-01-16 11:37:49 +01:00
|
|
|
// Send privileges
|
|
|
|
SendPlayerPrivileges(peer_id);
|
2011-05-23 21:40:25 +02:00
|
|
|
|
[Patch 2/4] Network rework: packet writing, sending and cleanups
NetworkPacket.cpp:
* Remove some deprecated functions, we must use streaming interface
* m_data converted from u8* to std::vector<u8>
* Add an exporter to forge packet to Connection object
* implement operator << std::wstring. n
* implement operator << std::string
* dynamic resize when write packet content.
* fix string writing and performances.
* create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Reliability
* Transmit channel
* Implement putRawString for some ugly char (_INIT packet), and use it.
* Many packet read and write migrated
* Implement oldForgePacket to interface writing with current connection
* fix U8/char/bool writing
* fix string writing and performances.
* add some missing functions
* Use v3s16 read instead of reading x,y,z separately
* Add irr::video::SColor support into packets
* Add some missing handlers
* Add a template function to increase offset
* Throw a serialization error on packet reading (must be improved)
PacketFactories:
* Create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Create ClientCommandFactory, used by server to get useful informations about packet processing (sending).
Client.cpp:
* implement NetworkPacket ::Send interface.
* Move packet handlers to a dedicated file
* Remove Client::Send(SharedBuffer)
Server.cpp:
* implement NetworkPacket ::Send interface.
* Rewrite all packets using NetworkPacket
* Move packet handlers to a dedicated file
* Remove Server::Send(SharedBuffer)
ClientIface.cpp:
* Remove sendToAll(SharedBuffer<u8>)
Connection.hpp rework:
* Remove duplicate include
* Remove duplicate negation
* Remove a useless variable
* Improve code performance by using a m_peers_list instead of scanning m_peers map
* Remove Connection::Send(SharedBuffer)
* Fix useafterfree into NetworkPacket Sending
* Remove unused Connection::sendToAll
Test.cpp:
* Remove dead code
* Update tests to use NetworkPackets
Misc:
* add new wrappers to Send packets in client, using NetworkPacket
* Add NetworkPacket methods for Connection
* coding style fix
* dead code since changes cleanup
* Use v3s16 read instead of reading x,y,z separately in some packets
* Use different files to handle packets received by client and server
* Cleanup: Remove useless includes
ok @Zeno-
Tested by @Zeno- @VanessaE and @nerzhul on running servers
2015-01-16 11:37:49 +01:00
|
|
|
// Send inventory formspec
|
|
|
|
SendPlayerInventoryFormspec(peer_id);
|
2012-11-26 03:16:48 +01:00
|
|
|
|
[Patch 2/4] Network rework: packet writing, sending and cleanups
NetworkPacket.cpp:
* Remove some deprecated functions, we must use streaming interface
* m_data converted from u8* to std::vector<u8>
* Add an exporter to forge packet to Connection object
* implement operator << std::wstring. n
* implement operator << std::string
* dynamic resize when write packet content.
* fix string writing and performances.
* create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Reliability
* Transmit channel
* Implement putRawString for some ugly char (_INIT packet), and use it.
* Many packet read and write migrated
* Implement oldForgePacket to interface writing with current connection
* fix U8/char/bool writing
* fix string writing and performances.
* add some missing functions
* Use v3s16 read instead of reading x,y,z separately
* Add irr::video::SColor support into packets
* Add some missing handlers
* Add a template function to increase offset
* Throw a serialization error on packet reading (must be improved)
PacketFactories:
* Create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Create ClientCommandFactory, used by server to get useful informations about packet processing (sending).
Client.cpp:
* implement NetworkPacket ::Send interface.
* Move packet handlers to a dedicated file
* Remove Client::Send(SharedBuffer)
Server.cpp:
* implement NetworkPacket ::Send interface.
* Rewrite all packets using NetworkPacket
* Move packet handlers to a dedicated file
* Remove Server::Send(SharedBuffer)
ClientIface.cpp:
* Remove sendToAll(SharedBuffer<u8>)
Connection.hpp rework:
* Remove duplicate include
* Remove duplicate negation
* Remove a useless variable
* Improve code performance by using a m_peers_list instead of scanning m_peers map
* Remove Connection::Send(SharedBuffer)
* Fix useafterfree into NetworkPacket Sending
* Remove unused Connection::sendToAll
Test.cpp:
* Remove dead code
* Update tests to use NetworkPackets
Misc:
* add new wrappers to Send packets in client, using NetworkPacket
* Add NetworkPacket methods for Connection
* coding style fix
* dead code since changes cleanup
* Use v3s16 read instead of reading x,y,z separately in some packets
* Use different files to handle packets received by client and server
* Cleanup: Remove useless includes
ok @Zeno-
Tested by @Zeno- @VanessaE and @nerzhul on running servers
2015-01-16 11:37:49 +01:00
|
|
|
// Send inventory
|
2015-03-04 15:27:53 +01:00
|
|
|
SendInventory(playersao);
|
2011-11-29 16:15:18 +01:00
|
|
|
|
2017-04-07 05:32:50 +02:00
|
|
|
// Send HP or death screen
|
|
|
|
if (playersao->isDead())
|
|
|
|
SendDeathscreen(peer_id, false, v3f(0,0,0));
|
|
|
|
else
|
2018-03-28 17:05:18 +02:00
|
|
|
SendPlayerHPOrDie(playersao,
|
|
|
|
PlayerHPChangeReason(PlayerHPChangeReason::SET_HP));
|
2012-01-12 06:10:39 +01:00
|
|
|
|
[Patch 2/4] Network rework: packet writing, sending and cleanups
NetworkPacket.cpp:
* Remove some deprecated functions, we must use streaming interface
* m_data converted from u8* to std::vector<u8>
* Add an exporter to forge packet to Connection object
* implement operator << std::wstring. n
* implement operator << std::string
* dynamic resize when write packet content.
* fix string writing and performances.
* create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Reliability
* Transmit channel
* Implement putRawString for some ugly char (_INIT packet), and use it.
* Many packet read and write migrated
* Implement oldForgePacket to interface writing with current connection
* fix U8/char/bool writing
* fix string writing and performances.
* add some missing functions
* Use v3s16 read instead of reading x,y,z separately
* Add irr::video::SColor support into packets
* Add some missing handlers
* Add a template function to increase offset
* Throw a serialization error on packet reading (must be improved)
PacketFactories:
* Create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Create ClientCommandFactory, used by server to get useful informations about packet processing (sending).
Client.cpp:
* implement NetworkPacket ::Send interface.
* Move packet handlers to a dedicated file
* Remove Client::Send(SharedBuffer)
Server.cpp:
* implement NetworkPacket ::Send interface.
* Rewrite all packets using NetworkPacket
* Move packet handlers to a dedicated file
* Remove Server::Send(SharedBuffer)
ClientIface.cpp:
* Remove sendToAll(SharedBuffer<u8>)
Connection.hpp rework:
* Remove duplicate include
* Remove duplicate negation
* Remove a useless variable
* Improve code performance by using a m_peers_list instead of scanning m_peers map
* Remove Connection::Send(SharedBuffer)
* Fix useafterfree into NetworkPacket Sending
* Remove unused Connection::sendToAll
Test.cpp:
* Remove dead code
* Update tests to use NetworkPackets
Misc:
* add new wrappers to Send packets in client, using NetworkPacket
* Add NetworkPacket methods for Connection
* coding style fix
* dead code since changes cleanup
* Use v3s16 read instead of reading x,y,z separately in some packets
* Use different files to handle packets received by client and server
* Cleanup: Remove useless includes
ok @Zeno-
Tested by @Zeno- @VanessaE and @nerzhul on running servers
2015-01-16 11:37:49 +01:00
|
|
|
// Send Breath
|
2017-01-01 16:13:01 +01:00
|
|
|
SendPlayerBreath(playersao);
|
2011-05-29 20:11:16 +02:00
|
|
|
|
2017-09-30 12:00:05 +02:00
|
|
|
Address addr = getPeerAddress(player->getPeerId());
|
[Patch 2/4] Network rework: packet writing, sending and cleanups
NetworkPacket.cpp:
* Remove some deprecated functions, we must use streaming interface
* m_data converted from u8* to std::vector<u8>
* Add an exporter to forge packet to Connection object
* implement operator << std::wstring. n
* implement operator << std::string
* dynamic resize when write packet content.
* fix string writing and performances.
* create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Reliability
* Transmit channel
* Implement putRawString for some ugly char (_INIT packet), and use it.
* Many packet read and write migrated
* Implement oldForgePacket to interface writing with current connection
* fix U8/char/bool writing
* fix string writing and performances.
* add some missing functions
* Use v3s16 read instead of reading x,y,z separately
* Add irr::video::SColor support into packets
* Add some missing handlers
* Add a template function to increase offset
* Throw a serialization error on packet reading (must be improved)
PacketFactories:
* Create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Create ClientCommandFactory, used by server to get useful informations about packet processing (sending).
Client.cpp:
* implement NetworkPacket ::Send interface.
* Move packet handlers to a dedicated file
* Remove Client::Send(SharedBuffer)
Server.cpp:
* implement NetworkPacket ::Send interface.
* Rewrite all packets using NetworkPacket
* Move packet handlers to a dedicated file
* Remove Server::Send(SharedBuffer)
ClientIface.cpp:
* Remove sendToAll(SharedBuffer<u8>)
Connection.hpp rework:
* Remove duplicate include
* Remove duplicate negation
* Remove a useless variable
* Improve code performance by using a m_peers_list instead of scanning m_peers map
* Remove Connection::Send(SharedBuffer)
* Fix useafterfree into NetworkPacket Sending
* Remove unused Connection::sendToAll
Test.cpp:
* Remove dead code
* Update tests to use NetworkPackets
Misc:
* add new wrappers to Send packets in client, using NetworkPacket
* Add NetworkPacket methods for Connection
* coding style fix
* dead code since changes cleanup
* Use v3s16 read instead of reading x,y,z separately in some packets
* Use different files to handle packets received by client and server
* Cleanup: Remove useless includes
ok @Zeno-
Tested by @Zeno- @VanessaE and @nerzhul on running servers
2015-01-16 11:37:49 +01:00
|
|
|
std::string ip_str = addr.serializeString();
|
|
|
|
actionstream<<player->getName() <<" [" << ip_str << "] joins game. " << std::endl;
|
|
|
|
/*
|
|
|
|
Print out action
|
|
|
|
*/
|
|
|
|
{
|
2016-10-08 16:42:09 +02:00
|
|
|
const std::vector<std::string> &names = m_clients.getPlayerNames();
|
2015-01-12 17:01:41 +01:00
|
|
|
|
2016-10-08 16:42:09 +02:00
|
|
|
actionstream << player->getName() << " joins game. List of players: ";
|
[Patch 2/4] Network rework: packet writing, sending and cleanups
NetworkPacket.cpp:
* Remove some deprecated functions, we must use streaming interface
* m_data converted from u8* to std::vector<u8>
* Add an exporter to forge packet to Connection object
* implement operator << std::wstring. n
* implement operator << std::string
* dynamic resize when write packet content.
* fix string writing and performances.
* create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Reliability
* Transmit channel
* Implement putRawString for some ugly char (_INIT packet), and use it.
* Many packet read and write migrated
* Implement oldForgePacket to interface writing with current connection
* fix U8/char/bool writing
* fix string writing and performances.
* add some missing functions
* Use v3s16 read instead of reading x,y,z separately
* Add irr::video::SColor support into packets
* Add some missing handlers
* Add a template function to increase offset
* Throw a serialization error on packet reading (must be improved)
PacketFactories:
* Create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Create ClientCommandFactory, used by server to get useful informations about packet processing (sending).
Client.cpp:
* implement NetworkPacket ::Send interface.
* Move packet handlers to a dedicated file
* Remove Client::Send(SharedBuffer)
Server.cpp:
* implement NetworkPacket ::Send interface.
* Rewrite all packets using NetworkPacket
* Move packet handlers to a dedicated file
* Remove Server::Send(SharedBuffer)
ClientIface.cpp:
* Remove sendToAll(SharedBuffer<u8>)
Connection.hpp rework:
* Remove duplicate include
* Remove duplicate negation
* Remove a useless variable
* Improve code performance by using a m_peers_list instead of scanning m_peers map
* Remove Connection::Send(SharedBuffer)
* Fix useafterfree into NetworkPacket Sending
* Remove unused Connection::sendToAll
Test.cpp:
* Remove dead code
* Update tests to use NetworkPackets
Misc:
* add new wrappers to Send packets in client, using NetworkPacket
* Add NetworkPacket methods for Connection
* coding style fix
* dead code since changes cleanup
* Use v3s16 read instead of reading x,y,z separately in some packets
* Use different files to handle packets received by client and server
* Cleanup: Remove useless includes
ok @Zeno-
Tested by @Zeno- @VanessaE and @nerzhul on running servers
2015-01-16 11:37:49 +01:00
|
|
|
|
2017-08-14 01:06:12 +02:00
|
|
|
for (const std::string &name : names) {
|
|
|
|
actionstream << name << " ";
|
[Patch 2/4] Network rework: packet writing, sending and cleanups
NetworkPacket.cpp:
* Remove some deprecated functions, we must use streaming interface
* m_data converted from u8* to std::vector<u8>
* Add an exporter to forge packet to Connection object
* implement operator << std::wstring. n
* implement operator << std::string
* dynamic resize when write packet content.
* fix string writing and performances.
* create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Reliability
* Transmit channel
* Implement putRawString for some ugly char (_INIT packet), and use it.
* Many packet read and write migrated
* Implement oldForgePacket to interface writing with current connection
* fix U8/char/bool writing
* fix string writing and performances.
* add some missing functions
* Use v3s16 read instead of reading x,y,z separately
* Add irr::video::SColor support into packets
* Add some missing handlers
* Add a template function to increase offset
* Throw a serialization error on packet reading (must be improved)
PacketFactories:
* Create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Create ClientCommandFactory, used by server to get useful informations about packet processing (sending).
Client.cpp:
* implement NetworkPacket ::Send interface.
* Move packet handlers to a dedicated file
* Remove Client::Send(SharedBuffer)
Server.cpp:
* implement NetworkPacket ::Send interface.
* Rewrite all packets using NetworkPacket
* Move packet handlers to a dedicated file
* Remove Server::Send(SharedBuffer)
ClientIface.cpp:
* Remove sendToAll(SharedBuffer<u8>)
Connection.hpp rework:
* Remove duplicate include
* Remove duplicate negation
* Remove a useless variable
* Improve code performance by using a m_peers_list instead of scanning m_peers map
* Remove Connection::Send(SharedBuffer)
* Fix useafterfree into NetworkPacket Sending
* Remove unused Connection::sendToAll
Test.cpp:
* Remove dead code
* Update tests to use NetworkPackets
Misc:
* add new wrappers to Send packets in client, using NetworkPacket
* Add NetworkPacket methods for Connection
* coding style fix
* dead code since changes cleanup
* Use v3s16 read instead of reading x,y,z separately in some packets
* Use different files to handle packets received by client and server
* Cleanup: Remove useless includes
ok @Zeno-
Tested by @Zeno- @VanessaE and @nerzhul on running servers
2015-01-16 11:37:49 +01:00
|
|
|
}
|
2015-01-12 17:01:41 +01:00
|
|
|
|
[Patch 2/4] Network rework: packet writing, sending and cleanups
NetworkPacket.cpp:
* Remove some deprecated functions, we must use streaming interface
* m_data converted from u8* to std::vector<u8>
* Add an exporter to forge packet to Connection object
* implement operator << std::wstring. n
* implement operator << std::string
* dynamic resize when write packet content.
* fix string writing and performances.
* create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Reliability
* Transmit channel
* Implement putRawString for some ugly char (_INIT packet), and use it.
* Many packet read and write migrated
* Implement oldForgePacket to interface writing with current connection
* fix U8/char/bool writing
* fix string writing and performances.
* add some missing functions
* Use v3s16 read instead of reading x,y,z separately
* Add irr::video::SColor support into packets
* Add some missing handlers
* Add a template function to increase offset
* Throw a serialization error on packet reading (must be improved)
PacketFactories:
* Create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Create ClientCommandFactory, used by server to get useful informations about packet processing (sending).
Client.cpp:
* implement NetworkPacket ::Send interface.
* Move packet handlers to a dedicated file
* Remove Client::Send(SharedBuffer)
Server.cpp:
* implement NetworkPacket ::Send interface.
* Rewrite all packets using NetworkPacket
* Move packet handlers to a dedicated file
* Remove Server::Send(SharedBuffer)
ClientIface.cpp:
* Remove sendToAll(SharedBuffer<u8>)
Connection.hpp rework:
* Remove duplicate include
* Remove duplicate negation
* Remove a useless variable
* Improve code performance by using a m_peers_list instead of scanning m_peers map
* Remove Connection::Send(SharedBuffer)
* Fix useafterfree into NetworkPacket Sending
* Remove unused Connection::sendToAll
Test.cpp:
* Remove dead code
* Update tests to use NetworkPackets
Misc:
* add new wrappers to Send packets in client, using NetworkPacket
* Add NetworkPacket methods for Connection
* coding style fix
* dead code since changes cleanup
* Use v3s16 read instead of reading x,y,z separately in some packets
* Use different files to handle packets received by client and server
* Cleanup: Remove useless includes
ok @Zeno-
Tested by @Zeno- @VanessaE and @nerzhul on running servers
2015-01-16 11:37:49 +01:00
|
|
|
actionstream << player->getName() <<std::endl;
|
|
|
|
}
|
|
|
|
return playersao;
|
2015-01-12 17:01:41 +01:00
|
|
|
}
|
|
|
|
|
[Patch 2/4] Network rework: packet writing, sending and cleanups
NetworkPacket.cpp:
* Remove some deprecated functions, we must use streaming interface
* m_data converted from u8* to std::vector<u8>
* Add an exporter to forge packet to Connection object
* implement operator << std::wstring. n
* implement operator << std::string
* dynamic resize when write packet content.
* fix string writing and performances.
* create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Reliability
* Transmit channel
* Implement putRawString for some ugly char (_INIT packet), and use it.
* Many packet read and write migrated
* Implement oldForgePacket to interface writing with current connection
* fix U8/char/bool writing
* fix string writing and performances.
* add some missing functions
* Use v3s16 read instead of reading x,y,z separately
* Add irr::video::SColor support into packets
* Add some missing handlers
* Add a template function to increase offset
* Throw a serialization error on packet reading (must be improved)
PacketFactories:
* Create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Create ClientCommandFactory, used by server to get useful informations about packet processing (sending).
Client.cpp:
* implement NetworkPacket ::Send interface.
* Move packet handlers to a dedicated file
* Remove Client::Send(SharedBuffer)
Server.cpp:
* implement NetworkPacket ::Send interface.
* Rewrite all packets using NetworkPacket
* Move packet handlers to a dedicated file
* Remove Server::Send(SharedBuffer)
ClientIface.cpp:
* Remove sendToAll(SharedBuffer<u8>)
Connection.hpp rework:
* Remove duplicate include
* Remove duplicate negation
* Remove a useless variable
* Improve code performance by using a m_peers_list instead of scanning m_peers map
* Remove Connection::Send(SharedBuffer)
* Fix useafterfree into NetworkPacket Sending
* Remove unused Connection::sendToAll
Test.cpp:
* Remove dead code
* Update tests to use NetworkPackets
Misc:
* add new wrappers to Send packets in client, using NetworkPacket
* Add NetworkPacket methods for Connection
* coding style fix
* dead code since changes cleanup
* Use v3s16 read instead of reading x,y,z separately in some packets
* Use different files to handle packets received by client and server
* Cleanup: Remove useless includes
ok @Zeno-
Tested by @Zeno- @VanessaE and @nerzhul on running servers
2015-01-16 11:37:49 +01:00
|
|
|
inline void Server::handleCommand(NetworkPacket* pkt)
|
2015-01-12 17:01:41 +01:00
|
|
|
{
|
|
|
|
const ToServerCommandHandler& opHandle = toServerCommandTable[pkt->getCommand()];
|
|
|
|
(this->*opHandle.handler)(pkt);
|
|
|
|
}
|
|
|
|
|
2015-03-31 10:35:51 +02:00
|
|
|
void Server::ProcessData(NetworkPacket *pkt)
|
2015-01-12 17:01:41 +01:00
|
|
|
{
|
|
|
|
// Environment is locked first.
|
2015-04-07 12:13:12 +02:00
|
|
|
MutexAutoLock envlock(m_env_mutex);
|
2015-01-12 17:01:41 +01:00
|
|
|
|
|
|
|
ScopeProfiler sp(g_profiler, "Server::ProcessData");
|
2015-03-31 10:35:51 +02:00
|
|
|
u32 peer_id = pkt->getPeerId();
|
2015-01-12 17:01:41 +01:00
|
|
|
|
|
|
|
try {
|
|
|
|
Address address = getPeerAddress(peer_id);
|
|
|
|
std::string addr_s = address.serializeString();
|
|
|
|
|
|
|
|
if(m_banmanager->isIpBanned(addr_s)) {
|
|
|
|
std::string ban_name = m_banmanager->getBanName(addr_s);
|
|
|
|
infostream << "Server: A banned client tried to connect from "
|
|
|
|
<< addr_s << "; banned name was "
|
|
|
|
<< ban_name << std::endl;
|
|
|
|
// This actually doesn't seem to transfer to the client
|
2015-03-13 16:35:21 +01:00
|
|
|
DenyAccess_Legacy(peer_id, L"Your ip is banned. Banned name was "
|
2015-07-07 05:55:07 +02:00
|
|
|
+ utf8_to_wide(ban_name));
|
2015-01-12 17:01:41 +01:00
|
|
|
return;
|
2012-03-24 18:01:26 +01:00
|
|
|
}
|
|
|
|
}
|
2015-01-12 17:01:41 +01:00
|
|
|
catch(con::PeerNotFoundException &e) {
|
|
|
|
/*
|
|
|
|
* no peer for this packet found
|
|
|
|
* most common reason is peer timeout, e.g. peer didn't
|
|
|
|
* respond for some time, your server was overloaded or
|
|
|
|
* things like that.
|
|
|
|
*/
|
2015-03-31 10:35:51 +02:00
|
|
|
infostream << "Server::ProcessData(): Canceling: peer "
|
2015-01-12 17:01:41 +01:00
|
|
|
<< peer_id << " not found" << std::endl;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
2015-03-31 10:35:51 +02:00
|
|
|
ToServerCommand command = (ToServerCommand) pkt->getCommand();
|
2015-01-12 17:01:41 +01:00
|
|
|
|
|
|
|
// Command must be handled into ToServerCommandHandler
|
|
|
|
if (command >= TOSERVER_NUM_MSG_TYPES) {
|
|
|
|
infostream << "Server: Ignoring unknown command "
|
|
|
|
<< command << std::endl;
|
2015-06-24 22:45:52 +02:00
|
|
|
return;
|
2012-06-01 19:51:15 +02:00
|
|
|
}
|
|
|
|
|
2015-01-12 17:01:41 +01:00
|
|
|
if (toServerCommandTable[command].state == TOSERVER_STATE_NOT_CONNECTED) {
|
2015-03-31 10:35:51 +02:00
|
|
|
handleCommand(pkt);
|
2015-01-12 17:01:41 +01:00
|
|
|
return;
|
|
|
|
}
|
2012-07-26 21:06:45 +02:00
|
|
|
|
2015-01-12 17:01:41 +01:00
|
|
|
u8 peer_ser_ver = getClient(peer_id, CS_InitDone)->serialization_version;
|
2012-07-26 21:06:45 +02:00
|
|
|
|
2015-01-12 17:01:41 +01:00
|
|
|
if(peer_ser_ver == SER_FMT_VER_INVALID) {
|
|
|
|
errorstream << "Server::ProcessData(): Cancelling: Peer"
|
|
|
|
" serialization format invalid or not initialized."
|
|
|
|
" Skipping incoming command=" << command << std::endl;
|
|
|
|
return;
|
2012-07-26 21:06:45 +02:00
|
|
|
}
|
2015-01-12 17:01:41 +01:00
|
|
|
|
|
|
|
/* Handle commands related to client startup */
|
|
|
|
if (toServerCommandTable[command].state == TOSERVER_STATE_STARTUP) {
|
2015-03-31 10:35:51 +02:00
|
|
|
handleCommand(pkt);
|
2015-01-12 17:01:41 +01:00
|
|
|
return;
|
2012-07-22 16:10:58 +02:00
|
|
|
}
|
|
|
|
|
2015-01-12 17:01:41 +01:00
|
|
|
if (m_clients.getClientState(peer_id) < CS_Active) {
|
|
|
|
if (command == TOSERVER_PLAYERPOS) return;
|
|
|
|
|
|
|
|
errorstream << "Got packet command: " << command << " for peer id "
|
|
|
|
<< peer_id << " but client isn't active yet. Dropping packet "
|
|
|
|
<< std::endl;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-03-31 10:35:51 +02:00
|
|
|
handleCommand(pkt);
|
2015-05-08 06:51:03 +02:00
|
|
|
} catch (SendFailedException &e) {
|
2015-01-12 17:01:41 +01:00
|
|
|
errorstream << "Server::ProcessData(): SendFailedException: "
|
|
|
|
<< "what=" << e.what()
|
|
|
|
<< std::endl;
|
2015-05-08 06:51:03 +02:00
|
|
|
} catch (PacketError &e) {
|
|
|
|
actionstream << "Server::ProcessData(): PacketError: "
|
|
|
|
<< "what=" << e.what()
|
|
|
|
<< std::endl;
|
2010-11-27 00:02:21 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-08-11 04:09:45 +02:00
|
|
|
void Server::setTimeOfDay(u32 time)
|
|
|
|
{
|
|
|
|
m_env->setTimeOfDay(time);
|
|
|
|
m_time_of_day_send_timer = 0;
|
|
|
|
}
|
|
|
|
|
2011-02-23 01:49:57 +01:00
|
|
|
void Server::onMapEditEvent(MapEditEvent *event)
|
2010-11-27 00:02:21 +01:00
|
|
|
{
|
2018-06-15 08:41:40 +02:00
|
|
|
if (m_ignore_map_edit_events_area.contains(event->getArea()))
|
2012-03-29 00:22:08 +02:00
|
|
|
return;
|
2011-02-23 01:49:57 +01:00
|
|
|
MapEditEvent *e = event->clone();
|
2015-03-04 17:48:07 +01:00
|
|
|
m_unsent_map_edit_queue.push(e);
|
2010-11-27 00:02:21 +01:00
|
|
|
}
|
|
|
|
|
2011-12-06 14:21:56 +01:00
|
|
|
Inventory* Server::getInventory(const InventoryLocation &loc)
|
|
|
|
{
|
2015-01-12 17:01:41 +01:00
|
|
|
switch (loc.type) {
|
2011-12-06 14:21:56 +01:00
|
|
|
case InventoryLocation::UNDEFINED:
|
2012-01-12 06:10:39 +01:00
|
|
|
case InventoryLocation::CURRENT_PLAYER:
|
2015-01-12 17:01:41 +01:00
|
|
|
break;
|
2011-12-06 14:21:56 +01:00
|
|
|
case InventoryLocation::PLAYER:
|
|
|
|
{
|
2017-08-14 01:06:12 +02:00
|
|
|
RemotePlayer *player = m_env->getPlayer(loc.name.c_str());
|
2011-12-06 14:21:56 +01:00
|
|
|
if(!player)
|
|
|
|
return NULL;
|
2012-03-19 03:04:16 +01:00
|
|
|
PlayerSAO *playersao = player->getPlayerSAO();
|
|
|
|
if(!playersao)
|
|
|
|
return NULL;
|
|
|
|
return playersao->getInventory();
|
2011-12-06 14:21:56 +01:00
|
|
|
}
|
2015-01-12 17:01:41 +01:00
|
|
|
break;
|
2011-12-06 14:21:56 +01:00
|
|
|
case InventoryLocation::NODEMETA:
|
|
|
|
{
|
|
|
|
NodeMetadata *meta = m_env->getMap().getNodeMetadata(loc.p);
|
|
|
|
if(!meta)
|
|
|
|
return NULL;
|
|
|
|
return meta->getInventory();
|
|
|
|
}
|
2015-01-12 17:01:41 +01:00
|
|
|
break;
|
2012-07-24 19:57:17 +02:00
|
|
|
case InventoryLocation::DETACHED:
|
|
|
|
{
|
|
|
|
if(m_detached_inventories.count(loc.name) == 0)
|
|
|
|
return NULL;
|
|
|
|
return m_detached_inventories[loc.name];
|
|
|
|
}
|
2015-01-12 17:01:41 +01:00
|
|
|
break;
|
2011-12-06 14:21:56 +01:00
|
|
|
default:
|
2015-03-06 11:21:51 +01:00
|
|
|
sanity_check(false); // abort
|
2015-03-04 15:27:53 +01:00
|
|
|
break;
|
2011-12-06 14:21:56 +01:00
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
2018-12-04 20:37:48 +01:00
|
|
|
|
2015-03-24 09:36:54 +01:00
|
|
|
void Server::setInventoryModified(const InventoryLocation &loc, bool playerSend)
|
2011-12-06 14:21:56 +01:00
|
|
|
{
|
|
|
|
switch(loc.type){
|
|
|
|
case InventoryLocation::UNDEFINED:
|
2015-01-12 17:01:41 +01:00
|
|
|
break;
|
2011-12-06 14:21:56 +01:00
|
|
|
case InventoryLocation::PLAYER:
|
|
|
|
{
|
2015-03-24 09:36:54 +01:00
|
|
|
if (!playerSend)
|
|
|
|
return;
|
|
|
|
|
2017-08-14 00:44:45 +02:00
|
|
|
RemotePlayer *player = m_env->getPlayer(loc.name.c_str());
|
2016-10-08 10:38:04 +02:00
|
|
|
|
|
|
|
if (!player)
|
2012-03-19 03:04:16 +01:00
|
|
|
return;
|
2016-10-08 10:38:04 +02:00
|
|
|
|
2012-03-19 03:04:16 +01:00
|
|
|
PlayerSAO *playersao = player->getPlayerSAO();
|
|
|
|
if(!playersao)
|
2011-12-06 14:21:56 +01:00
|
|
|
return;
|
2015-03-04 11:29:34 +01:00
|
|
|
|
2015-03-04 15:27:53 +01:00
|
|
|
SendInventory(playersao);
|
2011-12-06 14:21:56 +01:00
|
|
|
}
|
2015-01-12 17:01:41 +01:00
|
|
|
break;
|
2011-12-06 14:21:56 +01:00
|
|
|
case InventoryLocation::NODEMETA:
|
|
|
|
{
|
2018-12-04 20:37:48 +01:00
|
|
|
MapEditEvent event;
|
|
|
|
event.type = MEET_BLOCK_NODE_METADATA_CHANGED;
|
|
|
|
event.p = loc.p;
|
|
|
|
m_env->getMap().dispatchEvent(&event);
|
2011-12-06 14:21:56 +01:00
|
|
|
}
|
2015-01-12 17:01:41 +01:00
|
|
|
break;
|
2012-07-24 19:57:17 +02:00
|
|
|
case InventoryLocation::DETACHED:
|
|
|
|
{
|
2014-01-31 00:24:00 +01:00
|
|
|
sendDetachedInventory(loc.name,PEER_ID_INEXISTENT);
|
2012-07-24 19:57:17 +02:00
|
|
|
}
|
2015-01-12 17:01:41 +01:00
|
|
|
break;
|
2011-12-06 14:21:56 +01:00
|
|
|
default:
|
2015-03-06 11:21:51 +01:00
|
|
|
sanity_check(false); // abort
|
2015-03-04 15:27:53 +01:00
|
|
|
break;
|
2011-12-06 14:21:56 +01:00
|
|
|
}
|
|
|
|
}
|
2011-04-04 14:13:19 +02:00
|
|
|
|
2014-01-31 00:24:00 +01:00
|
|
|
void Server::SetBlocksNotSent(std::map<v3s16, MapBlock *>& block)
|
|
|
|
{
|
2017-09-27 23:48:06 +02:00
|
|
|
std::vector<session_t> clients = m_clients.getClientIDs();
|
2015-04-07 12:13:12 +02:00
|
|
|
m_clients.lock();
|
2014-01-31 00:24:00 +01:00
|
|
|
// Set the modified blocks unsent for all the clients
|
2017-09-27 23:48:06 +02:00
|
|
|
for (const session_t client_id : clients) {
|
2017-08-14 01:06:12 +02:00
|
|
|
if (RemoteClient *client = m_clients.lockedGetClientNoEx(client_id))
|
2014-01-31 00:24:00 +01:00
|
|
|
client->SetBlocksNotSent(block);
|
2015-03-04 16:30:24 +01:00
|
|
|
}
|
2015-04-07 12:13:12 +02:00
|
|
|
m_clients.unlock();
|
2014-01-31 00:24:00 +01:00
|
|
|
}
|
|
|
|
|
2010-11-27 00:02:21 +01:00
|
|
|
void Server::peerAdded(con::Peer *peer)
|
|
|
|
{
|
2012-03-11 03:15:45 +01:00
|
|
|
verbosestream<<"Server::peerAdded(): peer->id="
|
2010-11-27 00:02:21 +01:00
|
|
|
<<peer->id<<std::endl;
|
2012-11-26 03:16:48 +01:00
|
|
|
|
2017-08-15 09:30:31 +02:00
|
|
|
m_peer_change_queue.push(con::PeerChange(con::PEER_ADDED, peer->id, false));
|
2010-11-27 00:02:21 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void Server::deletingPeer(con::Peer *peer, bool timeout)
|
|
|
|
{
|
2012-03-11 03:15:45 +01:00
|
|
|
verbosestream<<"Server::deletingPeer(): peer->id="
|
2010-11-27 00:02:21 +01:00
|
|
|
<<peer->id<<", timeout="<<timeout<<std::endl;
|
2012-11-26 03:16:48 +01:00
|
|
|
|
2014-06-28 08:02:38 +02:00
|
|
|
m_clients.event(peer->id, CSE_Disconnect);
|
2017-08-15 09:30:31 +02:00
|
|
|
m_peer_change_queue.push(con::PeerChange(con::PEER_REMOVED, peer->id, timeout));
|
2010-11-27 00:02:21 +01:00
|
|
|
}
|
|
|
|
|
2017-09-27 19:47:36 +02:00
|
|
|
bool Server::getClientConInfo(session_t peer_id, con::rtt_stat_type type, float* retval)
|
2014-02-13 20:17:42 +01:00
|
|
|
{
|
2017-08-24 08:28:54 +02:00
|
|
|
*retval = m_con->getPeerStat(peer_id,type);
|
2017-08-14 01:06:12 +02:00
|
|
|
return *retval != -1;
|
2014-02-13 20:17:42 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
bool Server::getClientInfo(
|
2017-09-27 19:47:36 +02:00
|
|
|
session_t peer_id,
|
2014-02-13 20:17:42 +01:00
|
|
|
ClientState* state,
|
|
|
|
u32* uptime,
|
|
|
|
u8* ser_vers,
|
|
|
|
u16* prot_vers,
|
|
|
|
u8* major,
|
|
|
|
u8* minor,
|
|
|
|
u8* patch,
|
|
|
|
std::string* vers_string
|
|
|
|
)
|
|
|
|
{
|
|
|
|
*state = m_clients.getClientState(peer_id);
|
2015-04-07 12:13:12 +02:00
|
|
|
m_clients.lock();
|
2014-06-28 08:02:38 +02:00
|
|
|
RemoteClient* client = m_clients.lockedGetClientNoEx(peer_id, CS_Invalid);
|
2014-02-13 20:17:42 +01:00
|
|
|
|
2017-08-15 09:30:31 +02:00
|
|
|
if (!client) {
|
2015-04-07 12:13:12 +02:00
|
|
|
m_clients.unlock();
|
2014-02-13 20:17:42 +01:00
|
|
|
return false;
|
2014-06-28 08:02:38 +02:00
|
|
|
}
|
2014-02-13 20:17:42 +01:00
|
|
|
|
|
|
|
*uptime = client->uptime();
|
|
|
|
*ser_vers = client->serialization_version;
|
|
|
|
*prot_vers = client->net_proto_version;
|
|
|
|
|
|
|
|
*major = client->getMajor();
|
|
|
|
*minor = client->getMinor();
|
|
|
|
*patch = client->getPatch();
|
|
|
|
*vers_string = client->getPatch();
|
|
|
|
|
2015-04-07 12:13:12 +02:00
|
|
|
m_clients.unlock();
|
2014-02-13 20:17:42 +01:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2014-01-31 00:24:00 +01:00
|
|
|
void Server::handlePeerChanges()
|
|
|
|
{
|
2017-08-14 01:06:12 +02:00
|
|
|
while(!m_peer_change_queue.empty())
|
2014-01-31 00:24:00 +01:00
|
|
|
{
|
2015-03-04 17:48:07 +01:00
|
|
|
con::PeerChange c = m_peer_change_queue.front();
|
|
|
|
m_peer_change_queue.pop();
|
2014-01-31 00:24:00 +01:00
|
|
|
|
|
|
|
verbosestream<<"Server: Handling peer change: "
|
|
|
|
<<"id="<<c.peer_id<<", timeout="<<c.timeout
|
|
|
|
<<std::endl;
|
|
|
|
|
|
|
|
switch(c.type)
|
|
|
|
{
|
|
|
|
case con::PEER_ADDED:
|
|
|
|
m_clients.CreateClient(c.peer_id);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case con::PEER_REMOVED:
|
|
|
|
DeleteClient(c.peer_id, c.timeout?CDR_TIMEOUT:CDR_LEAVE);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
2015-03-06 11:21:51 +01:00
|
|
|
FATAL_ERROR("Invalid peer change event received!");
|
2014-01-31 00:24:00 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2011-04-21 18:35:17 +02:00
|
|
|
|
2015-07-25 07:43:32 +02:00
|
|
|
void Server::printToConsoleOnly(const std::string &text)
|
|
|
|
{
|
|
|
|
if (m_admin_chat) {
|
|
|
|
m_admin_chat->outgoing_queue.push_back(
|
|
|
|
new ChatEventChat("", utf8_to_wide(text)));
|
|
|
|
} else {
|
2015-11-08 22:23:16 +01:00
|
|
|
std::cout << text << std::endl;
|
2015-07-25 07:43:32 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-09-26 00:11:20 +02:00
|
|
|
void Server::Send(NetworkPacket *pkt)
|
[Patch 2/4] Network rework: packet writing, sending and cleanups
NetworkPacket.cpp:
* Remove some deprecated functions, we must use streaming interface
* m_data converted from u8* to std::vector<u8>
* Add an exporter to forge packet to Connection object
* implement operator << std::wstring. n
* implement operator << std::string
* dynamic resize when write packet content.
* fix string writing and performances.
* create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Reliability
* Transmit channel
* Implement putRawString for some ugly char (_INIT packet), and use it.
* Many packet read and write migrated
* Implement oldForgePacket to interface writing with current connection
* fix U8/char/bool writing
* fix string writing and performances.
* add some missing functions
* Use v3s16 read instead of reading x,y,z separately
* Add irr::video::SColor support into packets
* Add some missing handlers
* Add a template function to increase offset
* Throw a serialization error on packet reading (must be improved)
PacketFactories:
* Create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Create ClientCommandFactory, used by server to get useful informations about packet processing (sending).
Client.cpp:
* implement NetworkPacket ::Send interface.
* Move packet handlers to a dedicated file
* Remove Client::Send(SharedBuffer)
Server.cpp:
* implement NetworkPacket ::Send interface.
* Rewrite all packets using NetworkPacket
* Move packet handlers to a dedicated file
* Remove Server::Send(SharedBuffer)
ClientIface.cpp:
* Remove sendToAll(SharedBuffer<u8>)
Connection.hpp rework:
* Remove duplicate include
* Remove duplicate negation
* Remove a useless variable
* Improve code performance by using a m_peers_list instead of scanning m_peers map
* Remove Connection::Send(SharedBuffer)
* Fix useafterfree into NetworkPacket Sending
* Remove unused Connection::sendToAll
Test.cpp:
* Remove dead code
* Update tests to use NetworkPackets
Misc:
* add new wrappers to Send packets in client, using NetworkPacket
* Add NetworkPacket methods for Connection
* coding style fix
* dead code since changes cleanup
* Use v3s16 read instead of reading x,y,z separately in some packets
* Use different files to handle packets received by client and server
* Cleanup: Remove useless includes
ok @Zeno-
Tested by @Zeno- @VanessaE and @nerzhul on running servers
2015-01-16 11:37:49 +01:00
|
|
|
{
|
2017-09-26 00:11:20 +02:00
|
|
|
Send(pkt->getPeerId(), pkt);
|
|
|
|
}
|
|
|
|
|
2017-09-27 19:47:36 +02:00
|
|
|
void Server::Send(session_t peer_id, NetworkPacket *pkt)
|
2017-09-26 00:11:20 +02:00
|
|
|
{
|
|
|
|
m_clients.send(peer_id,
|
[Patch 2/4] Network rework: packet writing, sending and cleanups
NetworkPacket.cpp:
* Remove some deprecated functions, we must use streaming interface
* m_data converted from u8* to std::vector<u8>
* Add an exporter to forge packet to Connection object
* implement operator << std::wstring. n
* implement operator << std::string
* dynamic resize when write packet content.
* fix string writing and performances.
* create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Reliability
* Transmit channel
* Implement putRawString for some ugly char (_INIT packet), and use it.
* Many packet read and write migrated
* Implement oldForgePacket to interface writing with current connection
* fix U8/char/bool writing
* fix string writing and performances.
* add some missing functions
* Use v3s16 read instead of reading x,y,z separately
* Add irr::video::SColor support into packets
* Add some missing handlers
* Add a template function to increase offset
* Throw a serialization error on packet reading (must be improved)
PacketFactories:
* Create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Create ClientCommandFactory, used by server to get useful informations about packet processing (sending).
Client.cpp:
* implement NetworkPacket ::Send interface.
* Move packet handlers to a dedicated file
* Remove Client::Send(SharedBuffer)
Server.cpp:
* implement NetworkPacket ::Send interface.
* Rewrite all packets using NetworkPacket
* Move packet handlers to a dedicated file
* Remove Server::Send(SharedBuffer)
ClientIface.cpp:
* Remove sendToAll(SharedBuffer<u8>)
Connection.hpp rework:
* Remove duplicate include
* Remove duplicate negation
* Remove a useless variable
* Improve code performance by using a m_peers_list instead of scanning m_peers map
* Remove Connection::Send(SharedBuffer)
* Fix useafterfree into NetworkPacket Sending
* Remove unused Connection::sendToAll
Test.cpp:
* Remove dead code
* Update tests to use NetworkPackets
Misc:
* add new wrappers to Send packets in client, using NetworkPacket
* Add NetworkPacket methods for Connection
* coding style fix
* dead code since changes cleanup
* Use v3s16 read instead of reading x,y,z separately in some packets
* Use different files to handle packets received by client and server
* Cleanup: Remove useless includes
ok @Zeno-
Tested by @Zeno- @VanessaE and @nerzhul on running servers
2015-01-16 11:37:49 +01:00
|
|
|
clientCommandFactoryTable[pkt->getCommand()].channel,
|
|
|
|
pkt,
|
|
|
|
clientCommandFactoryTable[pkt->getCommand()].reliable);
|
|
|
|
}
|
|
|
|
|
2017-09-27 19:47:36 +02:00
|
|
|
void Server::SendMovement(session_t peer_id)
|
2013-02-08 21:54:01 +01:00
|
|
|
{
|
|
|
|
std::ostringstream os(std::ios_base::binary);
|
|
|
|
|
2015-03-13 22:01:49 +01:00
|
|
|
NetworkPacket pkt(TOCLIENT_MOVEMENT, 12 * sizeof(float), peer_id);
|
|
|
|
|
|
|
|
pkt << g_settings->getFloat("movement_acceleration_default");
|
|
|
|
pkt << g_settings->getFloat("movement_acceleration_air");
|
|
|
|
pkt << g_settings->getFloat("movement_acceleration_fast");
|
|
|
|
pkt << g_settings->getFloat("movement_speed_walk");
|
|
|
|
pkt << g_settings->getFloat("movement_speed_crouch");
|
|
|
|
pkt << g_settings->getFloat("movement_speed_fast");
|
|
|
|
pkt << g_settings->getFloat("movement_speed_climb");
|
|
|
|
pkt << g_settings->getFloat("movement_speed_jump");
|
|
|
|
pkt << g_settings->getFloat("movement_liquid_fluidity");
|
|
|
|
pkt << g_settings->getFloat("movement_liquid_fluidity_smooth");
|
|
|
|
pkt << g_settings->getFloat("movement_liquid_sink");
|
|
|
|
pkt << g_settings->getFloat("movement_gravity");
|
2013-02-08 21:54:01 +01:00
|
|
|
|
2015-03-13 22:01:49 +01:00
|
|
|
Send(&pkt);
|
2013-02-08 21:54:01 +01:00
|
|
|
}
|
|
|
|
|
2018-03-28 17:05:18 +02:00
|
|
|
void Server::SendPlayerHPOrDie(PlayerSAO *playersao, const PlayerHPChangeReason &reason)
|
2015-07-10 21:58:32 +02:00
|
|
|
{
|
|
|
|
if (!g_settings->getBool("enable_damage"))
|
|
|
|
return;
|
|
|
|
|
2017-09-27 19:47:36 +02:00
|
|
|
session_t peer_id = playersao->getPeerID();
|
2015-07-10 21:58:32 +02:00
|
|
|
bool is_alive = playersao->getHP() > 0;
|
|
|
|
|
|
|
|
if (is_alive)
|
|
|
|
SendPlayerHP(peer_id);
|
|
|
|
else
|
2018-03-28 17:05:18 +02:00
|
|
|
DiePlayer(peer_id, reason);
|
2015-07-10 21:58:32 +02:00
|
|
|
}
|
|
|
|
|
2017-09-27 19:47:36 +02:00
|
|
|
void Server::SendHP(session_t peer_id, u16 hp)
|
2011-04-21 18:35:17 +02:00
|
|
|
{
|
2015-03-13 22:01:49 +01:00
|
|
|
NetworkPacket pkt(TOCLIENT_HP, 1, peer_id);
|
|
|
|
pkt << hp;
|
|
|
|
Send(&pkt);
|
2011-04-21 18:35:17 +02:00
|
|
|
}
|
|
|
|
|
2017-09-27 19:47:36 +02:00
|
|
|
void Server::SendBreath(session_t peer_id, u16 breath)
|
2013-07-19 19:50:33 +02:00
|
|
|
{
|
2015-03-13 22:01:49 +01:00
|
|
|
NetworkPacket pkt(TOCLIENT_BREATH, 2, peer_id);
|
|
|
|
pkt << (u16) breath;
|
|
|
|
Send(&pkt);
|
2013-07-19 19:50:33 +02:00
|
|
|
}
|
|
|
|
|
2017-09-27 19:47:36 +02:00
|
|
|
void Server::SendAccessDenied(session_t peer_id, AccessDeniedCode reason,
|
2015-07-17 16:40:41 +02:00
|
|
|
const std::string &custom_reason, bool reconnect)
|
2015-03-13 16:35:21 +01:00
|
|
|
{
|
2015-07-17 16:40:41 +02:00
|
|
|
assert(reason < SERVER_ACCESSDENIED_MAX);
|
2015-03-13 16:35:21 +01:00
|
|
|
|
|
|
|
NetworkPacket pkt(TOCLIENT_ACCESS_DENIED, 1, peer_id);
|
2015-07-17 16:40:41 +02:00
|
|
|
pkt << (u8)reason;
|
|
|
|
if (reason == SERVER_ACCESSDENIED_CUSTOM_STRING)
|
2015-03-13 16:35:21 +01:00
|
|
|
pkt << custom_reason;
|
2015-07-17 16:40:41 +02:00
|
|
|
else if (reason == SERVER_ACCESSDENIED_SHUTDOWN ||
|
|
|
|
reason == SERVER_ACCESSDENIED_CRASH)
|
|
|
|
pkt << custom_reason << (u8)reconnect;
|
2015-03-13 16:35:21 +01:00
|
|
|
Send(&pkt);
|
|
|
|
}
|
|
|
|
|
2017-09-27 19:47:36 +02:00
|
|
|
void Server::SendAccessDenied_Legacy(session_t peer_id,const std::wstring &reason)
|
2011-05-20 21:28:03 +02:00
|
|
|
{
|
2015-03-13 22:01:49 +01:00
|
|
|
NetworkPacket pkt(TOCLIENT_ACCESS_DENIED_LEGACY, 0, peer_id);
|
|
|
|
pkt << reason;
|
|
|
|
Send(&pkt);
|
2011-05-20 21:28:03 +02:00
|
|
|
}
|
|
|
|
|
2017-09-27 19:47:36 +02:00
|
|
|
void Server::SendDeathscreen(session_t peer_id, bool set_camera_point_target,
|
2014-01-31 00:24:00 +01:00
|
|
|
v3f camera_point_target)
|
2011-10-15 13:46:59 +02:00
|
|
|
{
|
2015-03-13 22:01:49 +01:00
|
|
|
NetworkPacket pkt(TOCLIENT_DEATHSCREEN, 1 + sizeof(v3f), peer_id);
|
|
|
|
pkt << set_camera_point_target << camera_point_target;
|
|
|
|
Send(&pkt);
|
2011-10-15 13:46:59 +02:00
|
|
|
}
|
|
|
|
|
2017-09-27 19:47:36 +02:00
|
|
|
void Server::SendItemDef(session_t peer_id,
|
2013-03-28 21:40:44 +01:00
|
|
|
IItemDefManager *itemdef, u16 protocol_version)
|
2011-11-15 00:00:16 +01:00
|
|
|
{
|
2015-03-13 22:01:49 +01:00
|
|
|
NetworkPacket pkt(TOCLIENT_ITEMDEF, 0, peer_id);
|
2011-11-15 00:00:16 +01:00
|
|
|
|
|
|
|
/*
|
|
|
|
u16 command
|
|
|
|
u32 length of the next item
|
2012-01-12 06:10:39 +01:00
|
|
|
zlib-compressed serialized ItemDefManager
|
2011-11-15 00:00:16 +01:00
|
|
|
*/
|
|
|
|
std::ostringstream tmp_os(std::ios::binary);
|
2013-03-28 21:40:44 +01:00
|
|
|
itemdef->serialize(tmp_os, protocol_version);
|
2012-01-12 06:10:39 +01:00
|
|
|
std::ostringstream tmp_os2(std::ios::binary);
|
|
|
|
compressZlib(tmp_os.str(), tmp_os2);
|
2015-03-13 22:01:49 +01:00
|
|
|
pkt.putLongString(tmp_os2.str());
|
2011-11-15 00:00:16 +01:00
|
|
|
|
|
|
|
// Make data buffer
|
[Patch 2/4] Network rework: packet writing, sending and cleanups
NetworkPacket.cpp:
* Remove some deprecated functions, we must use streaming interface
* m_data converted from u8* to std::vector<u8>
* Add an exporter to forge packet to Connection object
* implement operator << std::wstring. n
* implement operator << std::string
* dynamic resize when write packet content.
* fix string writing and performances.
* create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Reliability
* Transmit channel
* Implement putRawString for some ugly char (_INIT packet), and use it.
* Many packet read and write migrated
* Implement oldForgePacket to interface writing with current connection
* fix U8/char/bool writing
* fix string writing and performances.
* add some missing functions
* Use v3s16 read instead of reading x,y,z separately
* Add irr::video::SColor support into packets
* Add some missing handlers
* Add a template function to increase offset
* Throw a serialization error on packet reading (must be improved)
PacketFactories:
* Create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Create ClientCommandFactory, used by server to get useful informations about packet processing (sending).
Client.cpp:
* implement NetworkPacket ::Send interface.
* Move packet handlers to a dedicated file
* Remove Client::Send(SharedBuffer)
Server.cpp:
* implement NetworkPacket ::Send interface.
* Rewrite all packets using NetworkPacket
* Move packet handlers to a dedicated file
* Remove Server::Send(SharedBuffer)
ClientIface.cpp:
* Remove sendToAll(SharedBuffer<u8>)
Connection.hpp rework:
* Remove duplicate include
* Remove duplicate negation
* Remove a useless variable
* Improve code performance by using a m_peers_list instead of scanning m_peers map
* Remove Connection::Send(SharedBuffer)
* Fix useafterfree into NetworkPacket Sending
* Remove unused Connection::sendToAll
Test.cpp:
* Remove dead code
* Update tests to use NetworkPackets
Misc:
* add new wrappers to Send packets in client, using NetworkPacket
* Add NetworkPacket methods for Connection
* coding style fix
* dead code since changes cleanup
* Use v3s16 read instead of reading x,y,z separately in some packets
* Use different files to handle packets received by client and server
* Cleanup: Remove useless includes
ok @Zeno-
Tested by @Zeno- @VanessaE and @nerzhul on running servers
2015-01-16 11:37:49 +01:00
|
|
|
verbosestream << "Server: Sending item definitions to id(" << peer_id
|
2015-03-13 22:01:49 +01:00
|
|
|
<< "): size=" << pkt.getSize() << std::endl;
|
[Patch 2/4] Network rework: packet writing, sending and cleanups
NetworkPacket.cpp:
* Remove some deprecated functions, we must use streaming interface
* m_data converted from u8* to std::vector<u8>
* Add an exporter to forge packet to Connection object
* implement operator << std::wstring. n
* implement operator << std::string
* dynamic resize when write packet content.
* fix string writing and performances.
* create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Reliability
* Transmit channel
* Implement putRawString for some ugly char (_INIT packet), and use it.
* Many packet read and write migrated
* Implement oldForgePacket to interface writing with current connection
* fix U8/char/bool writing
* fix string writing and performances.
* add some missing functions
* Use v3s16 read instead of reading x,y,z separately
* Add irr::video::SColor support into packets
* Add some missing handlers
* Add a template function to increase offset
* Throw a serialization error on packet reading (must be improved)
PacketFactories:
* Create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Create ClientCommandFactory, used by server to get useful informations about packet processing (sending).
Client.cpp:
* implement NetworkPacket ::Send interface.
* Move packet handlers to a dedicated file
* Remove Client::Send(SharedBuffer)
Server.cpp:
* implement NetworkPacket ::Send interface.
* Rewrite all packets using NetworkPacket
* Move packet handlers to a dedicated file
* Remove Server::Send(SharedBuffer)
ClientIface.cpp:
* Remove sendToAll(SharedBuffer<u8>)
Connection.hpp rework:
* Remove duplicate include
* Remove duplicate negation
* Remove a useless variable
* Improve code performance by using a m_peers_list instead of scanning m_peers map
* Remove Connection::Send(SharedBuffer)
* Fix useafterfree into NetworkPacket Sending
* Remove unused Connection::sendToAll
Test.cpp:
* Remove dead code
* Update tests to use NetworkPackets
Misc:
* add new wrappers to Send packets in client, using NetworkPacket
* Add NetworkPacket methods for Connection
* coding style fix
* dead code since changes cleanup
* Use v3s16 read instead of reading x,y,z separately in some packets
* Use different files to handle packets received by client and server
* Cleanup: Remove useless includes
ok @Zeno-
Tested by @Zeno- @VanessaE and @nerzhul on running servers
2015-01-16 11:37:49 +01:00
|
|
|
|
2015-03-13 22:01:49 +01:00
|
|
|
Send(&pkt);
|
2011-11-15 19:32:56 +01:00
|
|
|
}
|
|
|
|
|
2017-09-27 19:47:36 +02:00
|
|
|
void Server::SendNodeDef(session_t peer_id,
|
2018-02-10 21:04:16 +01:00
|
|
|
const NodeDefManager *nodedef, u16 protocol_version)
|
2011-11-15 19:32:56 +01:00
|
|
|
{
|
2015-03-13 22:01:49 +01:00
|
|
|
NetworkPacket pkt(TOCLIENT_NODEDEF, 0, peer_id);
|
2011-11-15 19:32:56 +01:00
|
|
|
|
|
|
|
/*
|
|
|
|
u16 command
|
|
|
|
u32 length of the next item
|
2012-01-12 06:10:39 +01:00
|
|
|
zlib-compressed serialized NodeDefManager
|
2011-11-15 19:32:56 +01:00
|
|
|
*/
|
|
|
|
std::ostringstream tmp_os(std::ios::binary);
|
2012-11-26 22:59:03 +01:00
|
|
|
nodedef->serialize(tmp_os, protocol_version);
|
2012-01-12 06:10:39 +01:00
|
|
|
std::ostringstream tmp_os2(std::ios::binary);
|
|
|
|
compressZlib(tmp_os.str(), tmp_os2);
|
[Patch 2/4] Network rework: packet writing, sending and cleanups
NetworkPacket.cpp:
* Remove some deprecated functions, we must use streaming interface
* m_data converted from u8* to std::vector<u8>
* Add an exporter to forge packet to Connection object
* implement operator << std::wstring. n
* implement operator << std::string
* dynamic resize when write packet content.
* fix string writing and performances.
* create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Reliability
* Transmit channel
* Implement putRawString for some ugly char (_INIT packet), and use it.
* Many packet read and write migrated
* Implement oldForgePacket to interface writing with current connection
* fix U8/char/bool writing
* fix string writing and performances.
* add some missing functions
* Use v3s16 read instead of reading x,y,z separately
* Add irr::video::SColor support into packets
* Add some missing handlers
* Add a template function to increase offset
* Throw a serialization error on packet reading (must be improved)
PacketFactories:
* Create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Create ClientCommandFactory, used by server to get useful informations about packet processing (sending).
Client.cpp:
* implement NetworkPacket ::Send interface.
* Move packet handlers to a dedicated file
* Remove Client::Send(SharedBuffer)
Server.cpp:
* implement NetworkPacket ::Send interface.
* Rewrite all packets using NetworkPacket
* Move packet handlers to a dedicated file
* Remove Server::Send(SharedBuffer)
ClientIface.cpp:
* Remove sendToAll(SharedBuffer<u8>)
Connection.hpp rework:
* Remove duplicate include
* Remove duplicate negation
* Remove a useless variable
* Improve code performance by using a m_peers_list instead of scanning m_peers map
* Remove Connection::Send(SharedBuffer)
* Fix useafterfree into NetworkPacket Sending
* Remove unused Connection::sendToAll
Test.cpp:
* Remove dead code
* Update tests to use NetworkPackets
Misc:
* add new wrappers to Send packets in client, using NetworkPacket
* Add NetworkPacket methods for Connection
* coding style fix
* dead code since changes cleanup
* Use v3s16 read instead of reading x,y,z separately in some packets
* Use different files to handle packets received by client and server
* Cleanup: Remove useless includes
ok @Zeno-
Tested by @Zeno- @VanessaE and @nerzhul on running servers
2015-01-16 11:37:49 +01:00
|
|
|
|
2015-03-13 22:01:49 +01:00
|
|
|
pkt.putLongString(tmp_os2.str());
|
2011-11-15 19:32:56 +01:00
|
|
|
|
|
|
|
// Make data buffer
|
[Patch 2/4] Network rework: packet writing, sending and cleanups
NetworkPacket.cpp:
* Remove some deprecated functions, we must use streaming interface
* m_data converted from u8* to std::vector<u8>
* Add an exporter to forge packet to Connection object
* implement operator << std::wstring. n
* implement operator << std::string
* dynamic resize when write packet content.
* fix string writing and performances.
* create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Reliability
* Transmit channel
* Implement putRawString for some ugly char (_INIT packet), and use it.
* Many packet read and write migrated
* Implement oldForgePacket to interface writing with current connection
* fix U8/char/bool writing
* fix string writing and performances.
* add some missing functions
* Use v3s16 read instead of reading x,y,z separately
* Add irr::video::SColor support into packets
* Add some missing handlers
* Add a template function to increase offset
* Throw a serialization error on packet reading (must be improved)
PacketFactories:
* Create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Create ClientCommandFactory, used by server to get useful informations about packet processing (sending).
Client.cpp:
* implement NetworkPacket ::Send interface.
* Move packet handlers to a dedicated file
* Remove Client::Send(SharedBuffer)
Server.cpp:
* implement NetworkPacket ::Send interface.
* Rewrite all packets using NetworkPacket
* Move packet handlers to a dedicated file
* Remove Server::Send(SharedBuffer)
ClientIface.cpp:
* Remove sendToAll(SharedBuffer<u8>)
Connection.hpp rework:
* Remove duplicate include
* Remove duplicate negation
* Remove a useless variable
* Improve code performance by using a m_peers_list instead of scanning m_peers map
* Remove Connection::Send(SharedBuffer)
* Fix useafterfree into NetworkPacket Sending
* Remove unused Connection::sendToAll
Test.cpp:
* Remove dead code
* Update tests to use NetworkPackets
Misc:
* add new wrappers to Send packets in client, using NetworkPacket
* Add NetworkPacket methods for Connection
* coding style fix
* dead code since changes cleanup
* Use v3s16 read instead of reading x,y,z separately in some packets
* Use different files to handle packets received by client and server
* Cleanup: Remove useless includes
ok @Zeno-
Tested by @Zeno- @VanessaE and @nerzhul on running servers
2015-01-16 11:37:49 +01:00
|
|
|
verbosestream << "Server: Sending node definitions to id(" << peer_id
|
2015-03-13 22:01:49 +01:00
|
|
|
<< "): size=" << pkt.getSize() << std::endl;
|
[Patch 2/4] Network rework: packet writing, sending and cleanups
NetworkPacket.cpp:
* Remove some deprecated functions, we must use streaming interface
* m_data converted from u8* to std::vector<u8>
* Add an exporter to forge packet to Connection object
* implement operator << std::wstring. n
* implement operator << std::string
* dynamic resize when write packet content.
* fix string writing and performances.
* create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Reliability
* Transmit channel
* Implement putRawString for some ugly char (_INIT packet), and use it.
* Many packet read and write migrated
* Implement oldForgePacket to interface writing with current connection
* fix U8/char/bool writing
* fix string writing and performances.
* add some missing functions
* Use v3s16 read instead of reading x,y,z separately
* Add irr::video::SColor support into packets
* Add some missing handlers
* Add a template function to increase offset
* Throw a serialization error on packet reading (must be improved)
PacketFactories:
* Create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Create ClientCommandFactory, used by server to get useful informations about packet processing (sending).
Client.cpp:
* implement NetworkPacket ::Send interface.
* Move packet handlers to a dedicated file
* Remove Client::Send(SharedBuffer)
Server.cpp:
* implement NetworkPacket ::Send interface.
* Rewrite all packets using NetworkPacket
* Move packet handlers to a dedicated file
* Remove Server::Send(SharedBuffer)
ClientIface.cpp:
* Remove sendToAll(SharedBuffer<u8>)
Connection.hpp rework:
* Remove duplicate include
* Remove duplicate negation
* Remove a useless variable
* Improve code performance by using a m_peers_list instead of scanning m_peers map
* Remove Connection::Send(SharedBuffer)
* Fix useafterfree into NetworkPacket Sending
* Remove unused Connection::sendToAll
Test.cpp:
* Remove dead code
* Update tests to use NetworkPackets
Misc:
* add new wrappers to Send packets in client, using NetworkPacket
* Add NetworkPacket methods for Connection
* coding style fix
* dead code since changes cleanup
* Use v3s16 read instead of reading x,y,z separately in some packets
* Use different files to handle packets received by client and server
* Cleanup: Remove useless includes
ok @Zeno-
Tested by @Zeno- @VanessaE and @nerzhul on running servers
2015-01-16 11:37:49 +01:00
|
|
|
|
2015-03-13 22:01:49 +01:00
|
|
|
Send(&pkt);
|
2011-11-15 00:00:16 +01:00
|
|
|
}
|
|
|
|
|
2011-04-21 18:35:17 +02:00
|
|
|
/*
|
|
|
|
Non-static send methods
|
|
|
|
*/
|
|
|
|
|
2015-03-04 15:27:53 +01:00
|
|
|
void Server::SendInventory(PlayerSAO* playerSAO)
|
2010-11-27 00:02:21 +01:00
|
|
|
{
|
2015-03-04 15:27:53 +01:00
|
|
|
UpdateCrafting(playerSAO->getPlayer());
|
2011-11-29 22:18:20 +01:00
|
|
|
|
2010-12-22 15:30:23 +01:00
|
|
|
/*
|
2011-04-21 18:35:17 +02:00
|
|
|
Serialize it
|
2010-12-22 15:30:23 +01:00
|
|
|
*/
|
2011-04-21 18:35:17 +02:00
|
|
|
|
2015-03-13 22:01:49 +01:00
|
|
|
NetworkPacket pkt(TOCLIENT_INVENTORY, 0, playerSAO->getPeerID());
|
[Patch 2/4] Network rework: packet writing, sending and cleanups
NetworkPacket.cpp:
* Remove some deprecated functions, we must use streaming interface
* m_data converted from u8* to std::vector<u8>
* Add an exporter to forge packet to Connection object
* implement operator << std::wstring. n
* implement operator << std::string
* dynamic resize when write packet content.
* fix string writing and performances.
* create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Reliability
* Transmit channel
* Implement putRawString for some ugly char (_INIT packet), and use it.
* Many packet read and write migrated
* Implement oldForgePacket to interface writing with current connection
* fix U8/char/bool writing
* fix string writing and performances.
* add some missing functions
* Use v3s16 read instead of reading x,y,z separately
* Add irr::video::SColor support into packets
* Add some missing handlers
* Add a template function to increase offset
* Throw a serialization error on packet reading (must be improved)
PacketFactories:
* Create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Create ClientCommandFactory, used by server to get useful informations about packet processing (sending).
Client.cpp:
* implement NetworkPacket ::Send interface.
* Move packet handlers to a dedicated file
* Remove Client::Send(SharedBuffer)
Server.cpp:
* implement NetworkPacket ::Send interface.
* Rewrite all packets using NetworkPacket
* Move packet handlers to a dedicated file
* Remove Server::Send(SharedBuffer)
ClientIface.cpp:
* Remove sendToAll(SharedBuffer<u8>)
Connection.hpp rework:
* Remove duplicate include
* Remove duplicate negation
* Remove a useless variable
* Improve code performance by using a m_peers_list instead of scanning m_peers map
* Remove Connection::Send(SharedBuffer)
* Fix useafterfree into NetworkPacket Sending
* Remove unused Connection::sendToAll
Test.cpp:
* Remove dead code
* Update tests to use NetworkPackets
Misc:
* add new wrappers to Send packets in client, using NetworkPacket
* Add NetworkPacket methods for Connection
* coding style fix
* dead code since changes cleanup
* Use v3s16 read instead of reading x,y,z separately in some packets
* Use different files to handle packets received by client and server
* Cleanup: Remove useless includes
ok @Zeno-
Tested by @Zeno- @VanessaE and @nerzhul on running servers
2015-01-16 11:37:49 +01:00
|
|
|
|
2011-04-21 18:35:17 +02:00
|
|
|
std::ostringstream os;
|
2015-03-04 15:27:53 +01:00
|
|
|
playerSAO->getInventory()->serialize(os);
|
2011-04-21 18:35:17 +02:00
|
|
|
|
|
|
|
std::string s = os.str();
|
2012-11-26 03:16:48 +01:00
|
|
|
|
2015-03-13 22:01:49 +01:00
|
|
|
pkt.putRawString(s.c_str(), s.size());
|
|
|
|
Send(&pkt);
|
2011-04-21 18:35:17 +02:00
|
|
|
}
|
|
|
|
|
2017-09-27 19:47:36 +02:00
|
|
|
void Server::SendChatMessage(session_t peer_id, const ChatMessage &message)
|
2011-04-21 18:35:17 +02:00
|
|
|
{
|
2017-07-16 10:47:31 +02:00
|
|
|
NetworkPacket pkt(TOCLIENT_CHAT_MESSAGE, 0, peer_id);
|
|
|
|
u8 version = 1;
|
|
|
|
u8 type = message.type;
|
|
|
|
pkt << version << type << std::wstring(L"") << message.message << message.timestamp;
|
|
|
|
|
2017-06-09 21:39:25 +02:00
|
|
|
if (peer_id != PEER_ID_INEXISTENT) {
|
2017-07-16 10:47:31 +02:00
|
|
|
RemotePlayer *player = m_env->getPlayer(peer_id);
|
|
|
|
if (!player)
|
|
|
|
return;
|
2012-11-26 03:16:48 +01:00
|
|
|
|
2017-09-12 19:48:09 +02:00
|
|
|
Send(&pkt);
|
2017-06-09 21:39:25 +02:00
|
|
|
} else {
|
2017-09-12 19:48:09 +02:00
|
|
|
m_clients.sendToAll(&pkt);
|
2014-01-31 00:24:00 +01:00
|
|
|
}
|
2011-04-21 18:35:17 +02:00
|
|
|
}
|
2013-04-14 00:20:22 +02:00
|
|
|
|
2017-09-27 19:47:36 +02:00
|
|
|
void Server::SendShowFormspecMessage(session_t peer_id, const std::string &formspec,
|
2014-02-27 21:12:59 +01:00
|
|
|
const std::string &formname)
|
2013-01-02 20:45:04 +01:00
|
|
|
{
|
2015-03-13 22:01:49 +01:00
|
|
|
NetworkPacket pkt(TOCLIENT_SHOW_FORMSPEC, 0 , peer_id);
|
2017-08-14 01:06:12 +02:00
|
|
|
if (formspec.empty()){
|
2016-11-22 17:15:22 +01:00
|
|
|
//the client should close the formspec
|
2019-01-21 09:53:09 +01:00
|
|
|
//but make sure there wasn't another one open in meantime
|
|
|
|
const auto it = m_formspec_state_data.find(peer_id);
|
|
|
|
if (it != m_formspec_state_data.end() && it->second == formname) {
|
|
|
|
m_formspec_state_data.erase(peer_id);
|
|
|
|
}
|
2016-11-22 17:15:22 +01:00
|
|
|
pkt.putLongString("");
|
|
|
|
} else {
|
2018-02-18 22:33:42 +01:00
|
|
|
m_formspec_state_data[peer_id] = formname;
|
2016-11-22 17:15:22 +01:00
|
|
|
pkt.putLongString(FORMSPEC_VERSION_STRING + formspec);
|
|
|
|
}
|
2015-03-13 22:01:49 +01:00
|
|
|
pkt << formname;
|
2013-01-02 20:45:04 +01:00
|
|
|
|
2015-03-13 22:01:49 +01:00
|
|
|
Send(&pkt);
|
2013-01-02 20:45:04 +01:00
|
|
|
}
|
2011-04-11 20:28:04 +02:00
|
|
|
|
2013-01-23 18:32:02 +01:00
|
|
|
// Spawns a particle on peer with peer_id
|
2017-09-27 19:47:36 +02:00
|
|
|
void Server::SendSpawnParticle(session_t peer_id, u16 protocol_version,
|
2017-01-14 16:48:49 +01:00
|
|
|
v3f pos, v3f velocity, v3f acceleration,
|
2013-04-14 00:20:22 +02:00
|
|
|
float expirationtime, float size, bool collisiondetection,
|
2018-09-08 01:38:35 +02:00
|
|
|
bool collision_removal, bool object_collision,
|
2017-01-14 16:48:49 +01:00
|
|
|
bool vertical, const std::string &texture,
|
|
|
|
const struct TileAnimationParams &animation, u8 glow)
|
2013-01-23 18:32:02 +01:00
|
|
|
{
|
2017-06-11 13:58:43 +02:00
|
|
|
static thread_local const float radius =
|
2017-05-17 10:37:28 +02:00
|
|
|
g_settings->getS16("max_block_send_distance") * MAP_BLOCKSIZE * BS;
|
|
|
|
|
2017-01-14 16:48:49 +01:00
|
|
|
if (peer_id == PEER_ID_INEXISTENT) {
|
2017-09-27 23:48:06 +02:00
|
|
|
std::vector<session_t> clients = m_clients.getClientIDs();
|
2017-05-17 10:37:28 +02:00
|
|
|
|
2017-09-27 23:48:06 +02:00
|
|
|
for (const session_t client_id : clients) {
|
2017-08-14 01:06:12 +02:00
|
|
|
RemotePlayer *player = m_env->getPlayer(client_id);
|
2017-01-14 16:48:49 +01:00
|
|
|
if (!player)
|
|
|
|
continue;
|
2017-05-17 10:37:28 +02:00
|
|
|
|
|
|
|
PlayerSAO *sao = player->getPlayerSAO();
|
|
|
|
if (!sao)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
// Do not send to distant clients
|
|
|
|
if (sao->getBasePosition().getDistanceFrom(pos * BS) > radius)
|
|
|
|
continue;
|
|
|
|
|
2017-08-14 01:06:12 +02:00
|
|
|
SendSpawnParticle(client_id, player->protocol_version,
|
2017-01-14 16:48:49 +01:00
|
|
|
pos, velocity, acceleration,
|
2018-09-08 01:38:35 +02:00
|
|
|
expirationtime, size, collisiondetection, collision_removal,
|
|
|
|
object_collision, vertical, texture, animation, glow);
|
2017-01-14 16:48:49 +01:00
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
2013-01-23 18:32:02 +01:00
|
|
|
|
2015-03-13 22:01:49 +01:00
|
|
|
NetworkPacket pkt(TOCLIENT_SPAWN_PARTICLE, 0, peer_id);
|
2013-01-23 18:32:02 +01:00
|
|
|
|
2015-03-13 22:01:49 +01:00
|
|
|
pkt << pos << velocity << acceleration << expirationtime
|
[Patch 2/4] Network rework: packet writing, sending and cleanups
NetworkPacket.cpp:
* Remove some deprecated functions, we must use streaming interface
* m_data converted from u8* to std::vector<u8>
* Add an exporter to forge packet to Connection object
* implement operator << std::wstring. n
* implement operator << std::string
* dynamic resize when write packet content.
* fix string writing and performances.
* create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Reliability
* Transmit channel
* Implement putRawString for some ugly char (_INIT packet), and use it.
* Many packet read and write migrated
* Implement oldForgePacket to interface writing with current connection
* fix U8/char/bool writing
* fix string writing and performances.
* add some missing functions
* Use v3s16 read instead of reading x,y,z separately
* Add irr::video::SColor support into packets
* Add some missing handlers
* Add a template function to increase offset
* Throw a serialization error on packet reading (must be improved)
PacketFactories:
* Create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Create ClientCommandFactory, used by server to get useful informations about packet processing (sending).
Client.cpp:
* implement NetworkPacket ::Send interface.
* Move packet handlers to a dedicated file
* Remove Client::Send(SharedBuffer)
Server.cpp:
* implement NetworkPacket ::Send interface.
* Rewrite all packets using NetworkPacket
* Move packet handlers to a dedicated file
* Remove Server::Send(SharedBuffer)
ClientIface.cpp:
* Remove sendToAll(SharedBuffer<u8>)
Connection.hpp rework:
* Remove duplicate include
* Remove duplicate negation
* Remove a useless variable
* Improve code performance by using a m_peers_list instead of scanning m_peers map
* Remove Connection::Send(SharedBuffer)
* Fix useafterfree into NetworkPacket Sending
* Remove unused Connection::sendToAll
Test.cpp:
* Remove dead code
* Update tests to use NetworkPackets
Misc:
* add new wrappers to Send packets in client, using NetworkPacket
* Add NetworkPacket methods for Connection
* coding style fix
* dead code since changes cleanup
* Use v3s16 read instead of reading x,y,z separately in some packets
* Use different files to handle packets received by client and server
* Cleanup: Remove useless includes
ok @Zeno-
Tested by @Zeno- @VanessaE and @nerzhul on running servers
2015-01-16 11:37:49 +01:00
|
|
|
<< size << collisiondetection;
|
2015-03-13 22:01:49 +01:00
|
|
|
pkt.putLongString(texture);
|
|
|
|
pkt << vertical;
|
2016-05-28 06:08:23 +02:00
|
|
|
pkt << collision_removal;
|
2017-01-14 16:48:49 +01:00
|
|
|
// This is horrible but required (why are there two ways to serialize pkts?)
|
|
|
|
std::ostringstream os(std::ios_base::binary);
|
|
|
|
animation.serialize(os, protocol_version);
|
|
|
|
pkt.putRawString(os.str());
|
|
|
|
pkt << glow;
|
2018-09-08 01:38:35 +02:00
|
|
|
pkt << object_collision;
|
2013-01-23 18:32:02 +01:00
|
|
|
|
2017-01-14 16:48:49 +01:00
|
|
|
Send(&pkt);
|
2013-01-23 18:32:02 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Adds a ParticleSpawner on peer with peer_id
|
2017-09-27 19:47:36 +02:00
|
|
|
void Server::SendAddParticleSpawner(session_t peer_id, u16 protocol_version,
|
2017-01-14 16:48:49 +01:00
|
|
|
u16 amount, float spawntime, v3f minpos, v3f maxpos,
|
2013-01-23 18:32:02 +01:00
|
|
|
v3f minvel, v3f maxvel, v3f minacc, v3f maxacc, float minexptime, float maxexptime,
|
2016-05-28 06:08:23 +02:00
|
|
|
float minsize, float maxsize, bool collisiondetection, bool collision_removal,
|
2018-09-08 01:38:35 +02:00
|
|
|
bool object_collision, u16 attached_id, bool vertical, const std::string &texture, u32 id,
|
2017-01-14 16:48:49 +01:00
|
|
|
const struct TileAnimationParams &animation, u8 glow)
|
2013-01-23 18:32:02 +01:00
|
|
|
{
|
2017-01-14 16:48:49 +01:00
|
|
|
if (peer_id == PEER_ID_INEXISTENT) {
|
|
|
|
// This sucks and should be replaced:
|
2017-09-27 23:48:06 +02:00
|
|
|
std::vector<session_t> clients = m_clients.getClientIDs();
|
|
|
|
for (const session_t client_id : clients) {
|
2017-08-14 01:06:12 +02:00
|
|
|
RemotePlayer *player = m_env->getPlayer(client_id);
|
2017-01-14 16:48:49 +01:00
|
|
|
if (!player)
|
|
|
|
continue;
|
2017-08-14 01:06:12 +02:00
|
|
|
SendAddParticleSpawner(client_id, player->protocol_version,
|
2017-01-14 16:48:49 +01:00
|
|
|
amount, spawntime, minpos, maxpos,
|
|
|
|
minvel, maxvel, minacc, maxacc, minexptime, maxexptime,
|
|
|
|
minsize, maxsize, collisiondetection, collision_removal,
|
2018-09-08 01:38:35 +02:00
|
|
|
object_collision, attached_id, vertical, texture, id,
|
|
|
|
animation, glow);
|
2017-01-14 16:48:49 +01:00
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
2013-01-23 18:32:02 +01:00
|
|
|
|
2015-03-13 22:01:49 +01:00
|
|
|
NetworkPacket pkt(TOCLIENT_ADD_PARTICLESPAWNER, 0, peer_id);
|
2013-01-23 18:32:02 +01:00
|
|
|
|
2015-03-13 22:01:49 +01:00
|
|
|
pkt << amount << spawntime << minpos << maxpos << minvel << maxvel
|
[Patch 2/4] Network rework: packet writing, sending and cleanups
NetworkPacket.cpp:
* Remove some deprecated functions, we must use streaming interface
* m_data converted from u8* to std::vector<u8>
* Add an exporter to forge packet to Connection object
* implement operator << std::wstring. n
* implement operator << std::string
* dynamic resize when write packet content.
* fix string writing and performances.
* create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Reliability
* Transmit channel
* Implement putRawString for some ugly char (_INIT packet), and use it.
* Many packet read and write migrated
* Implement oldForgePacket to interface writing with current connection
* fix U8/char/bool writing
* fix string writing and performances.
* add some missing functions
* Use v3s16 read instead of reading x,y,z separately
* Add irr::video::SColor support into packets
* Add some missing handlers
* Add a template function to increase offset
* Throw a serialization error on packet reading (must be improved)
PacketFactories:
* Create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Create ClientCommandFactory, used by server to get useful informations about packet processing (sending).
Client.cpp:
* implement NetworkPacket ::Send interface.
* Move packet handlers to a dedicated file
* Remove Client::Send(SharedBuffer)
Server.cpp:
* implement NetworkPacket ::Send interface.
* Rewrite all packets using NetworkPacket
* Move packet handlers to a dedicated file
* Remove Server::Send(SharedBuffer)
ClientIface.cpp:
* Remove sendToAll(SharedBuffer<u8>)
Connection.hpp rework:
* Remove duplicate include
* Remove duplicate negation
* Remove a useless variable
* Improve code performance by using a m_peers_list instead of scanning m_peers map
* Remove Connection::Send(SharedBuffer)
* Fix useafterfree into NetworkPacket Sending
* Remove unused Connection::sendToAll
Test.cpp:
* Remove dead code
* Update tests to use NetworkPackets
Misc:
* add new wrappers to Send packets in client, using NetworkPacket
* Add NetworkPacket methods for Connection
* coding style fix
* dead code since changes cleanup
* Use v3s16 read instead of reading x,y,z separately in some packets
* Use different files to handle packets received by client and server
* Cleanup: Remove useless includes
ok @Zeno-
Tested by @Zeno- @VanessaE and @nerzhul on running servers
2015-01-16 11:37:49 +01:00
|
|
|
<< minacc << maxacc << minexptime << maxexptime << minsize
|
|
|
|
<< maxsize << collisiondetection;
|
2013-01-23 18:32:02 +01:00
|
|
|
|
2015-03-13 22:01:49 +01:00
|
|
|
pkt.putLongString(texture);
|
[Patch 2/4] Network rework: packet writing, sending and cleanups
NetworkPacket.cpp:
* Remove some deprecated functions, we must use streaming interface
* m_data converted from u8* to std::vector<u8>
* Add an exporter to forge packet to Connection object
* implement operator << std::wstring. n
* implement operator << std::string
* dynamic resize when write packet content.
* fix string writing and performances.
* create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Reliability
* Transmit channel
* Implement putRawString for some ugly char (_INIT packet), and use it.
* Many packet read and write migrated
* Implement oldForgePacket to interface writing with current connection
* fix U8/char/bool writing
* fix string writing and performances.
* add some missing functions
* Use v3s16 read instead of reading x,y,z separately
* Add irr::video::SColor support into packets
* Add some missing handlers
* Add a template function to increase offset
* Throw a serialization error on packet reading (must be improved)
PacketFactories:
* Create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Create ClientCommandFactory, used by server to get useful informations about packet processing (sending).
Client.cpp:
* implement NetworkPacket ::Send interface.
* Move packet handlers to a dedicated file
* Remove Client::Send(SharedBuffer)
Server.cpp:
* implement NetworkPacket ::Send interface.
* Rewrite all packets using NetworkPacket
* Move packet handlers to a dedicated file
* Remove Server::Send(SharedBuffer)
ClientIface.cpp:
* Remove sendToAll(SharedBuffer<u8>)
Connection.hpp rework:
* Remove duplicate include
* Remove duplicate negation
* Remove a useless variable
* Improve code performance by using a m_peers_list instead of scanning m_peers map
* Remove Connection::Send(SharedBuffer)
* Fix useafterfree into NetworkPacket Sending
* Remove unused Connection::sendToAll
Test.cpp:
* Remove dead code
* Update tests to use NetworkPackets
Misc:
* add new wrappers to Send packets in client, using NetworkPacket
* Add NetworkPacket methods for Connection
* coding style fix
* dead code since changes cleanup
* Use v3s16 read instead of reading x,y,z separately in some packets
* Use different files to handle packets received by client and server
* Cleanup: Remove useless includes
ok @Zeno-
Tested by @Zeno- @VanessaE and @nerzhul on running servers
2015-01-16 11:37:49 +01:00
|
|
|
|
2015-03-13 22:01:49 +01:00
|
|
|
pkt << id << vertical;
|
2016-05-28 06:08:23 +02:00
|
|
|
pkt << collision_removal;
|
2016-08-04 22:09:21 +02:00
|
|
|
pkt << attached_id;
|
2017-01-14 16:48:49 +01:00
|
|
|
// This is horrible but required
|
|
|
|
std::ostringstream os(std::ios_base::binary);
|
|
|
|
animation.serialize(os, protocol_version);
|
|
|
|
pkt.putRawString(os.str());
|
|
|
|
pkt << glow;
|
2018-09-08 01:38:35 +02:00
|
|
|
pkt << object_collision;
|
[Patch 2/4] Network rework: packet writing, sending and cleanups
NetworkPacket.cpp:
* Remove some deprecated functions, we must use streaming interface
* m_data converted from u8* to std::vector<u8>
* Add an exporter to forge packet to Connection object
* implement operator << std::wstring. n
* implement operator << std::string
* dynamic resize when write packet content.
* fix string writing and performances.
* create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Reliability
* Transmit channel
* Implement putRawString for some ugly char (_INIT packet), and use it.
* Many packet read and write migrated
* Implement oldForgePacket to interface writing with current connection
* fix U8/char/bool writing
* fix string writing and performances.
* add some missing functions
* Use v3s16 read instead of reading x,y,z separately
* Add irr::video::SColor support into packets
* Add some missing handlers
* Add a template function to increase offset
* Throw a serialization error on packet reading (must be improved)
PacketFactories:
* Create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Create ClientCommandFactory, used by server to get useful informations about packet processing (sending).
Client.cpp:
* implement NetworkPacket ::Send interface.
* Move packet handlers to a dedicated file
* Remove Client::Send(SharedBuffer)
Server.cpp:
* implement NetworkPacket ::Send interface.
* Rewrite all packets using NetworkPacket
* Move packet handlers to a dedicated file
* Remove Server::Send(SharedBuffer)
ClientIface.cpp:
* Remove sendToAll(SharedBuffer<u8>)
Connection.hpp rework:
* Remove duplicate include
* Remove duplicate negation
* Remove a useless variable
* Improve code performance by using a m_peers_list instead of scanning m_peers map
* Remove Connection::Send(SharedBuffer)
* Fix useafterfree into NetworkPacket Sending
* Remove unused Connection::sendToAll
Test.cpp:
* Remove dead code
* Update tests to use NetworkPackets
Misc:
* add new wrappers to Send packets in client, using NetworkPacket
* Add NetworkPacket methods for Connection
* coding style fix
* dead code since changes cleanup
* Use v3s16 read instead of reading x,y,z separately in some packets
* Use different files to handle packets received by client and server
* Cleanup: Remove useless includes
ok @Zeno-
Tested by @Zeno- @VanessaE and @nerzhul on running servers
2015-01-16 11:37:49 +01:00
|
|
|
|
2017-01-14 16:48:49 +01:00
|
|
|
Send(&pkt);
|
2013-01-23 18:32:02 +01:00
|
|
|
}
|
|
|
|
|
2017-09-27 19:47:36 +02:00
|
|
|
void Server::SendDeleteParticleSpawner(session_t peer_id, u32 id)
|
2013-01-23 18:32:02 +01:00
|
|
|
{
|
2017-09-12 19:48:09 +02:00
|
|
|
NetworkPacket pkt(TOCLIENT_DELETE_PARTICLESPAWNER, 4, peer_id);
|
2013-01-23 18:32:02 +01:00
|
|
|
|
[Patch 2/4] Network rework: packet writing, sending and cleanups
NetworkPacket.cpp:
* Remove some deprecated functions, we must use streaming interface
* m_data converted from u8* to std::vector<u8>
* Add an exporter to forge packet to Connection object
* implement operator << std::wstring. n
* implement operator << std::string
* dynamic resize when write packet content.
* fix string writing and performances.
* create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Reliability
* Transmit channel
* Implement putRawString for some ugly char (_INIT packet), and use it.
* Many packet read and write migrated
* Implement oldForgePacket to interface writing with current connection
* fix U8/char/bool writing
* fix string writing and performances.
* add some missing functions
* Use v3s16 read instead of reading x,y,z separately
* Add irr::video::SColor support into packets
* Add some missing handlers
* Add a template function to increase offset
* Throw a serialization error on packet reading (must be improved)
PacketFactories:
* Create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Create ClientCommandFactory, used by server to get useful informations about packet processing (sending).
Client.cpp:
* implement NetworkPacket ::Send interface.
* Move packet handlers to a dedicated file
* Remove Client::Send(SharedBuffer)
Server.cpp:
* implement NetworkPacket ::Send interface.
* Rewrite all packets using NetworkPacket
* Move packet handlers to a dedicated file
* Remove Server::Send(SharedBuffer)
ClientIface.cpp:
* Remove sendToAll(SharedBuffer<u8>)
Connection.hpp rework:
* Remove duplicate include
* Remove duplicate negation
* Remove a useless variable
* Improve code performance by using a m_peers_list instead of scanning m_peers map
* Remove Connection::Send(SharedBuffer)
* Fix useafterfree into NetworkPacket Sending
* Remove unused Connection::sendToAll
Test.cpp:
* Remove dead code
* Update tests to use NetworkPackets
Misc:
* add new wrappers to Send packets in client, using NetworkPacket
* Add NetworkPacket methods for Connection
* coding style fix
* dead code since changes cleanup
* Use v3s16 read instead of reading x,y,z separately in some packets
* Use different files to handle packets received by client and server
* Cleanup: Remove useless includes
ok @Zeno-
Tested by @Zeno- @VanessaE and @nerzhul on running servers
2015-01-16 11:37:49 +01:00
|
|
|
// Ugly error in this packet
|
2017-09-12 19:48:09 +02:00
|
|
|
pkt << id;
|
2013-01-23 18:32:02 +01:00
|
|
|
|
2017-09-12 19:48:09 +02:00
|
|
|
if (peer_id != PEER_ID_INEXISTENT)
|
2015-03-13 22:01:49 +01:00
|
|
|
Send(&pkt);
|
2017-09-12 19:48:09 +02:00
|
|
|
else
|
2017-04-14 15:34:01 +02:00
|
|
|
m_clients.sendToAll(&pkt);
|
2014-01-31 00:24:00 +01:00
|
|
|
|
2013-01-23 18:32:02 +01:00
|
|
|
}
|
|
|
|
|
2017-09-27 19:47:36 +02:00
|
|
|
void Server::SendHUDAdd(session_t peer_id, u32 id, HudElement *form)
|
2013-04-11 20:23:38 +02:00
|
|
|
{
|
2015-03-13 22:01:49 +01:00
|
|
|
NetworkPacket pkt(TOCLIENT_HUDADD, 0 , peer_id);
|
2013-04-11 20:23:38 +02:00
|
|
|
|
2015-03-13 22:01:49 +01:00
|
|
|
pkt << id << (u8) form->type << form->pos << form->name << form->scale
|
[Patch 2/4] Network rework: packet writing, sending and cleanups
NetworkPacket.cpp:
* Remove some deprecated functions, we must use streaming interface
* m_data converted from u8* to std::vector<u8>
* Add an exporter to forge packet to Connection object
* implement operator << std::wstring. n
* implement operator << std::string
* dynamic resize when write packet content.
* fix string writing and performances.
* create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Reliability
* Transmit channel
* Implement putRawString for some ugly char (_INIT packet), and use it.
* Many packet read and write migrated
* Implement oldForgePacket to interface writing with current connection
* fix U8/char/bool writing
* fix string writing and performances.
* add some missing functions
* Use v3s16 read instead of reading x,y,z separately
* Add irr::video::SColor support into packets
* Add some missing handlers
* Add a template function to increase offset
* Throw a serialization error on packet reading (must be improved)
PacketFactories:
* Create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Create ClientCommandFactory, used by server to get useful informations about packet processing (sending).
Client.cpp:
* implement NetworkPacket ::Send interface.
* Move packet handlers to a dedicated file
* Remove Client::Send(SharedBuffer)
Server.cpp:
* implement NetworkPacket ::Send interface.
* Rewrite all packets using NetworkPacket
* Move packet handlers to a dedicated file
* Remove Server::Send(SharedBuffer)
ClientIface.cpp:
* Remove sendToAll(SharedBuffer<u8>)
Connection.hpp rework:
* Remove duplicate include
* Remove duplicate negation
* Remove a useless variable
* Improve code performance by using a m_peers_list instead of scanning m_peers map
* Remove Connection::Send(SharedBuffer)
* Fix useafterfree into NetworkPacket Sending
* Remove unused Connection::sendToAll
Test.cpp:
* Remove dead code
* Update tests to use NetworkPackets
Misc:
* add new wrappers to Send packets in client, using NetworkPacket
* Add NetworkPacket methods for Connection
* coding style fix
* dead code since changes cleanup
* Use v3s16 read instead of reading x,y,z separately in some packets
* Use different files to handle packets received by client and server
* Cleanup: Remove useless includes
ok @Zeno-
Tested by @Zeno- @VanessaE and @nerzhul on running servers
2015-01-16 11:37:49 +01:00
|
|
|
<< form->text << form->number << form->item << form->dir
|
|
|
|
<< form->align << form->offset << form->world_pos << form->size;
|
2013-04-11 20:23:38 +02:00
|
|
|
|
2015-03-13 22:01:49 +01:00
|
|
|
Send(&pkt);
|
2013-04-11 20:23:38 +02:00
|
|
|
}
|
|
|
|
|
2017-09-27 19:47:36 +02:00
|
|
|
void Server::SendHUDRemove(session_t peer_id, u32 id)
|
2013-04-11 20:23:38 +02:00
|
|
|
{
|
2015-03-13 22:01:49 +01:00
|
|
|
NetworkPacket pkt(TOCLIENT_HUDRM, 4, peer_id);
|
|
|
|
pkt << id;
|
|
|
|
Send(&pkt);
|
2013-04-11 20:23:38 +02:00
|
|
|
}
|
|
|
|
|
2017-09-27 19:47:36 +02:00
|
|
|
void Server::SendHUDChange(session_t peer_id, u32 id, HudElementStat stat, void *value)
|
2013-04-11 20:23:38 +02:00
|
|
|
{
|
2015-03-13 22:01:49 +01:00
|
|
|
NetworkPacket pkt(TOCLIENT_HUDCHANGE, 0, peer_id);
|
|
|
|
pkt << id << (u8) stat;
|
2013-04-11 20:23:38 +02:00
|
|
|
|
2013-04-14 00:20:22 +02:00
|
|
|
switch (stat) {
|
|
|
|
case HUD_STAT_POS:
|
|
|
|
case HUD_STAT_SCALE:
|
2013-04-22 11:53:55 +02:00
|
|
|
case HUD_STAT_ALIGN:
|
2013-04-23 01:47:59 +02:00
|
|
|
case HUD_STAT_OFFSET:
|
2015-03-13 22:01:49 +01:00
|
|
|
pkt << *(v2f *) value;
|
2013-04-14 00:20:22 +02:00
|
|
|
break;
|
|
|
|
case HUD_STAT_NAME:
|
|
|
|
case HUD_STAT_TEXT:
|
2015-03-13 22:01:49 +01:00
|
|
|
pkt << *(std::string *) value;
|
2013-04-14 00:20:22 +02:00
|
|
|
break;
|
2014-01-26 21:31:59 +01:00
|
|
|
case HUD_STAT_WORLD_POS:
|
2015-03-13 22:01:49 +01:00
|
|
|
pkt << *(v3f *) value;
|
2014-01-26 21:31:59 +01:00
|
|
|
break;
|
2014-04-28 23:41:27 +02:00
|
|
|
case HUD_STAT_SIZE:
|
2015-03-13 22:01:49 +01:00
|
|
|
pkt << *(v2s32 *) value;
|
2014-04-28 23:41:27 +02:00
|
|
|
break;
|
2013-04-14 00:20:22 +02:00
|
|
|
case HUD_STAT_NUMBER:
|
|
|
|
case HUD_STAT_ITEM:
|
|
|
|
case HUD_STAT_DIR:
|
|
|
|
default:
|
2015-03-13 22:01:49 +01:00
|
|
|
pkt << *(u32 *) value;
|
2013-04-14 00:20:22 +02:00
|
|
|
break;
|
|
|
|
}
|
2013-04-11 20:23:38 +02:00
|
|
|
|
2015-03-13 22:01:49 +01:00
|
|
|
Send(&pkt);
|
2013-04-11 20:23:38 +02:00
|
|
|
}
|
|
|
|
|
2017-09-27 19:47:36 +02:00
|
|
|
void Server::SendHUDSetFlags(session_t peer_id, u32 flags, u32 mask)
|
2013-04-24 12:52:46 +02:00
|
|
|
{
|
2015-03-13 22:01:49 +01:00
|
|
|
NetworkPacket pkt(TOCLIENT_HUD_SET_FLAGS, 4 + 4, peer_id);
|
2013-04-24 12:52:46 +02:00
|
|
|
|
2014-05-11 00:35:31 +02:00
|
|
|
flags &= ~(HUD_FLAG_HEALTHBAR_VISIBLE | HUD_FLAG_BREATHBAR_VISIBLE);
|
2013-04-24 12:52:46 +02:00
|
|
|
|
2015-03-13 22:01:49 +01:00
|
|
|
pkt << flags << mask;
|
[Patch 2/4] Network rework: packet writing, sending and cleanups
NetworkPacket.cpp:
* Remove some deprecated functions, we must use streaming interface
* m_data converted from u8* to std::vector<u8>
* Add an exporter to forge packet to Connection object
* implement operator << std::wstring. n
* implement operator << std::string
* dynamic resize when write packet content.
* fix string writing and performances.
* create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Reliability
* Transmit channel
* Implement putRawString for some ugly char (_INIT packet), and use it.
* Many packet read and write migrated
* Implement oldForgePacket to interface writing with current connection
* fix U8/char/bool writing
* fix string writing and performances.
* add some missing functions
* Use v3s16 read instead of reading x,y,z separately
* Add irr::video::SColor support into packets
* Add some missing handlers
* Add a template function to increase offset
* Throw a serialization error on packet reading (must be improved)
PacketFactories:
* Create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Create ClientCommandFactory, used by server to get useful informations about packet processing (sending).
Client.cpp:
* implement NetworkPacket ::Send interface.
* Move packet handlers to a dedicated file
* Remove Client::Send(SharedBuffer)
Server.cpp:
* implement NetworkPacket ::Send interface.
* Rewrite all packets using NetworkPacket
* Move packet handlers to a dedicated file
* Remove Server::Send(SharedBuffer)
ClientIface.cpp:
* Remove sendToAll(SharedBuffer<u8>)
Connection.hpp rework:
* Remove duplicate include
* Remove duplicate negation
* Remove a useless variable
* Improve code performance by using a m_peers_list instead of scanning m_peers map
* Remove Connection::Send(SharedBuffer)
* Fix useafterfree into NetworkPacket Sending
* Remove unused Connection::sendToAll
Test.cpp:
* Remove dead code
* Update tests to use NetworkPackets
Misc:
* add new wrappers to Send packets in client, using NetworkPacket
* Add NetworkPacket methods for Connection
* coding style fix
* dead code since changes cleanup
* Use v3s16 read instead of reading x,y,z separately in some packets
* Use different files to handle packets received by client and server
* Cleanup: Remove useless includes
ok @Zeno-
Tested by @Zeno- @VanessaE and @nerzhul on running servers
2015-01-16 11:37:49 +01:00
|
|
|
|
2015-03-13 22:01:49 +01:00
|
|
|
Send(&pkt);
|
2013-04-24 12:52:46 +02:00
|
|
|
}
|
|
|
|
|
2017-09-27 19:47:36 +02:00
|
|
|
void Server::SendHUDSetParam(session_t peer_id, u16 param, const std::string &value)
|
2013-05-04 02:08:52 +02:00
|
|
|
{
|
2015-03-13 22:01:49 +01:00
|
|
|
NetworkPacket pkt(TOCLIENT_HUD_SET_PARAM, 0, peer_id);
|
|
|
|
pkt << param << value;
|
|
|
|
Send(&pkt);
|
2011-04-21 18:35:17 +02:00
|
|
|
}
|
2010-12-25 00:54:39 +01:00
|
|
|
|
2017-09-27 19:47:36 +02:00
|
|
|
void Server::SendSetSky(session_t peer_id, const video::SColor &bgcolor,
|
2017-04-28 04:06:49 +02:00
|
|
|
const std::string &type, const std::vector<std::string> ¶ms,
|
|
|
|
bool &clouds)
|
2013-05-02 22:52:50 +02:00
|
|
|
{
|
2015-03-13 22:01:49 +01:00
|
|
|
NetworkPacket pkt(TOCLIENT_SET_SKY, 0, peer_id);
|
|
|
|
pkt << bgcolor << type << (u16) params.size();
|
2013-05-02 22:52:50 +02:00
|
|
|
|
2017-08-14 01:06:12 +02:00
|
|
|
for (const std::string ¶m : params)
|
|
|
|
pkt << param;
|
2013-05-02 22:52:50 +02:00
|
|
|
|
2017-04-28 04:06:49 +02:00
|
|
|
pkt << clouds;
|
|
|
|
|
2015-03-13 22:01:49 +01:00
|
|
|
Send(&pkt);
|
2013-05-02 22:52:50 +02:00
|
|
|
}
|
|
|
|
|
2018-03-08 22:19:25 +01:00
|
|
|
void Server::SendCloudParams(session_t peer_id, const CloudParams ¶ms)
|
2017-03-17 10:39:47 +01:00
|
|
|
{
|
|
|
|
NetworkPacket pkt(TOCLIENT_CLOUD_PARAMS, 0, peer_id);
|
2018-03-08 22:19:25 +01:00
|
|
|
pkt << params.density << params.color_bright << params.color_ambient
|
|
|
|
<< params.height << params.thickness << params.speed;
|
2017-03-17 10:39:47 +01:00
|
|
|
Send(&pkt);
|
|
|
|
}
|
|
|
|
|
2017-09-27 19:47:36 +02:00
|
|
|
void Server::SendOverrideDayNightRatio(session_t peer_id, bool do_override,
|
2013-10-18 10:53:19 +02:00
|
|
|
float ratio)
|
|
|
|
{
|
2015-03-13 22:01:49 +01:00
|
|
|
NetworkPacket pkt(TOCLIENT_OVERRIDE_DAY_NIGHT_RATIO,
|
[Patch 2/4] Network rework: packet writing, sending and cleanups
NetworkPacket.cpp:
* Remove some deprecated functions, we must use streaming interface
* m_data converted from u8* to std::vector<u8>
* Add an exporter to forge packet to Connection object
* implement operator << std::wstring. n
* implement operator << std::string
* dynamic resize when write packet content.
* fix string writing and performances.
* create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Reliability
* Transmit channel
* Implement putRawString for some ugly char (_INIT packet), and use it.
* Many packet read and write migrated
* Implement oldForgePacket to interface writing with current connection
* fix U8/char/bool writing
* fix string writing and performances.
* add some missing functions
* Use v3s16 read instead of reading x,y,z separately
* Add irr::video::SColor support into packets
* Add some missing handlers
* Add a template function to increase offset
* Throw a serialization error on packet reading (must be improved)
PacketFactories:
* Create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Create ClientCommandFactory, used by server to get useful informations about packet processing (sending).
Client.cpp:
* implement NetworkPacket ::Send interface.
* Move packet handlers to a dedicated file
* Remove Client::Send(SharedBuffer)
Server.cpp:
* implement NetworkPacket ::Send interface.
* Rewrite all packets using NetworkPacket
* Move packet handlers to a dedicated file
* Remove Server::Send(SharedBuffer)
ClientIface.cpp:
* Remove sendToAll(SharedBuffer<u8>)
Connection.hpp rework:
* Remove duplicate include
* Remove duplicate negation
* Remove a useless variable
* Improve code performance by using a m_peers_list instead of scanning m_peers map
* Remove Connection::Send(SharedBuffer)
* Fix useafterfree into NetworkPacket Sending
* Remove unused Connection::sendToAll
Test.cpp:
* Remove dead code
* Update tests to use NetworkPackets
Misc:
* add new wrappers to Send packets in client, using NetworkPacket
* Add NetworkPacket methods for Connection
* coding style fix
* dead code since changes cleanup
* Use v3s16 read instead of reading x,y,z separately in some packets
* Use different files to handle packets received by client and server
* Cleanup: Remove useless includes
ok @Zeno-
Tested by @Zeno- @VanessaE and @nerzhul on running servers
2015-01-16 11:37:49 +01:00
|
|
|
1 + 2, peer_id);
|
2013-10-18 10:53:19 +02:00
|
|
|
|
2015-03-13 22:01:49 +01:00
|
|
|
pkt << do_override << (u16) (ratio * 65535);
|
2013-10-18 10:53:19 +02:00
|
|
|
|
2015-03-13 22:01:49 +01:00
|
|
|
Send(&pkt);
|
2013-10-18 10:53:19 +02:00
|
|
|
}
|
|
|
|
|
2017-09-27 19:47:36 +02:00
|
|
|
void Server::SendTimeOfDay(session_t peer_id, u16 time, f32 time_speed)
|
2013-08-11 04:09:45 +02:00
|
|
|
{
|
2015-03-13 22:01:49 +01:00
|
|
|
NetworkPacket pkt(TOCLIENT_TIME_OF_DAY, 0, peer_id);
|
|
|
|
pkt << time << time_speed;
|
2013-08-11 04:09:45 +02:00
|
|
|
|
2014-01-31 00:24:00 +01:00
|
|
|
if (peer_id == PEER_ID_INEXISTENT) {
|
2017-04-14 15:34:01 +02:00
|
|
|
m_clients.sendToAll(&pkt);
|
2014-01-31 00:24:00 +01:00
|
|
|
}
|
|
|
|
else {
|
2015-03-13 22:01:49 +01:00
|
|
|
Send(&pkt);
|
2014-01-31 00:24:00 +01:00
|
|
|
}
|
2013-08-11 04:09:45 +02:00
|
|
|
}
|
|
|
|
|
2017-09-27 19:47:36 +02:00
|
|
|
void Server::SendPlayerHP(session_t peer_id)
|
2011-04-21 18:35:17 +02:00
|
|
|
{
|
2012-03-19 03:04:16 +01:00
|
|
|
PlayerSAO *playersao = getPlayerSAO(peer_id);
|
2016-05-08 11:02:11 +02:00
|
|
|
// In some rare case if the player is disconnected
|
2015-04-11 10:07:23 +02:00
|
|
|
// while Lua call l_punch, for example, this can be NULL
|
|
|
|
if (!playersao)
|
|
|
|
return;
|
|
|
|
|
2014-01-31 00:24:00 +01:00
|
|
|
SendHP(peer_id, playersao->getHP());
|
2014-04-28 23:41:27 +02:00
|
|
|
m_script->player_event(playersao,"health_changed");
|
2013-08-13 23:06:39 +02:00
|
|
|
|
|
|
|
// Send to other clients
|
2019-02-11 00:03:26 +01:00
|
|
|
std::string str = gob_cmd_punched(playersao->getHP());
|
2013-08-13 23:06:39 +02:00
|
|
|
ActiveObjectMessage aom(playersao->getId(), true, str);
|
2015-03-04 17:48:07 +01:00
|
|
|
playersao->m_messages_out.push(aom);
|
2011-04-21 18:35:17 +02:00
|
|
|
}
|
|
|
|
|
2017-01-01 16:13:01 +01:00
|
|
|
void Server::SendPlayerBreath(PlayerSAO *sao)
|
2013-07-19 19:50:33 +02:00
|
|
|
{
|
2017-01-01 16:13:01 +01:00
|
|
|
assert(sao);
|
2015-03-03 16:23:47 +01:00
|
|
|
|
2017-01-01 16:13:01 +01:00
|
|
|
m_script->player_event(sao, "breath_changed");
|
|
|
|
SendBreath(sao->getPeerID(), sao->getBreath());
|
2013-07-19 19:50:33 +02:00
|
|
|
}
|
|
|
|
|
2017-09-27 19:47:36 +02:00
|
|
|
void Server::SendMovePlayer(session_t peer_id)
|
2011-04-21 18:35:17 +02:00
|
|
|
{
|
2016-10-08 16:31:22 +02:00
|
|
|
RemotePlayer *player = m_env->getPlayer(peer_id);
|
2012-03-19 03:04:16 +01:00
|
|
|
assert(player);
|
2016-10-30 14:53:26 +01:00
|
|
|
PlayerSAO *sao = player->getPlayerSAO();
|
|
|
|
assert(sao);
|
2011-04-21 18:35:17 +02:00
|
|
|
|
2015-03-13 22:01:49 +01:00
|
|
|
NetworkPacket pkt(TOCLIENT_MOVE_PLAYER, sizeof(v3f) + sizeof(f32) * 2, peer_id);
|
2018-11-28 09:38:50 +01:00
|
|
|
pkt << sao->getBasePosition() << sao->getLookPitch() << sao->getRotation().Y;
|
2012-11-26 03:16:48 +01:00
|
|
|
|
2011-04-21 18:35:17 +02:00
|
|
|
{
|
2016-10-30 14:53:26 +01:00
|
|
|
v3f pos = sao->getBasePosition();
|
2015-03-13 22:01:49 +01:00
|
|
|
verbosestream << "Server: Sending TOCLIENT_MOVE_PLAYER"
|
|
|
|
<< " pos=(" << pos.X << "," << pos.Y << "," << pos.Z << ")"
|
2018-11-28 09:38:50 +01:00
|
|
|
<< " pitch=" << sao->getLookPitch()
|
|
|
|
<< " yaw=" << sao->getRotation().Y
|
2015-03-13 22:01:49 +01:00
|
|
|
<< std::endl;
|
2011-04-21 18:35:17 +02:00
|
|
|
}
|
|
|
|
|
2015-03-13 22:01:49 +01:00
|
|
|
Send(&pkt);
|
2011-04-21 18:35:17 +02:00
|
|
|
}
|
|
|
|
|
2017-09-27 19:47:36 +02:00
|
|
|
void Server::SendLocalPlayerAnimations(session_t peer_id, v2s32 animation_frames[4],
|
|
|
|
f32 animation_speed)
|
2014-01-08 13:47:53 +01:00
|
|
|
{
|
2015-03-13 22:01:49 +01:00
|
|
|
NetworkPacket pkt(TOCLIENT_LOCAL_PLAYER_ANIMATIONS, 0,
|
[Patch 2/4] Network rework: packet writing, sending and cleanups
NetworkPacket.cpp:
* Remove some deprecated functions, we must use streaming interface
* m_data converted from u8* to std::vector<u8>
* Add an exporter to forge packet to Connection object
* implement operator << std::wstring. n
* implement operator << std::string
* dynamic resize when write packet content.
* fix string writing and performances.
* create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Reliability
* Transmit channel
* Implement putRawString for some ugly char (_INIT packet), and use it.
* Many packet read and write migrated
* Implement oldForgePacket to interface writing with current connection
* fix U8/char/bool writing
* fix string writing and performances.
* add some missing functions
* Use v3s16 read instead of reading x,y,z separately
* Add irr::video::SColor support into packets
* Add some missing handlers
* Add a template function to increase offset
* Throw a serialization error on packet reading (must be improved)
PacketFactories:
* Create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Create ClientCommandFactory, used by server to get useful informations about packet processing (sending).
Client.cpp:
* implement NetworkPacket ::Send interface.
* Move packet handlers to a dedicated file
* Remove Client::Send(SharedBuffer)
Server.cpp:
* implement NetworkPacket ::Send interface.
* Rewrite all packets using NetworkPacket
* Move packet handlers to a dedicated file
* Remove Server::Send(SharedBuffer)
ClientIface.cpp:
* Remove sendToAll(SharedBuffer<u8>)
Connection.hpp rework:
* Remove duplicate include
* Remove duplicate negation
* Remove a useless variable
* Improve code performance by using a m_peers_list instead of scanning m_peers map
* Remove Connection::Send(SharedBuffer)
* Fix useafterfree into NetworkPacket Sending
* Remove unused Connection::sendToAll
Test.cpp:
* Remove dead code
* Update tests to use NetworkPackets
Misc:
* add new wrappers to Send packets in client, using NetworkPacket
* Add NetworkPacket methods for Connection
* coding style fix
* dead code since changes cleanup
* Use v3s16 read instead of reading x,y,z separately in some packets
* Use different files to handle packets received by client and server
* Cleanup: Remove useless includes
ok @Zeno-
Tested by @Zeno- @VanessaE and @nerzhul on running servers
2015-01-16 11:37:49 +01:00
|
|
|
peer_id);
|
2014-01-08 13:47:53 +01:00
|
|
|
|
2015-03-13 22:01:49 +01:00
|
|
|
pkt << animation_frames[0] << animation_frames[1] << animation_frames[2]
|
[Patch 2/4] Network rework: packet writing, sending and cleanups
NetworkPacket.cpp:
* Remove some deprecated functions, we must use streaming interface
* m_data converted from u8* to std::vector<u8>
* Add an exporter to forge packet to Connection object
* implement operator << std::wstring. n
* implement operator << std::string
* dynamic resize when write packet content.
* fix string writing and performances.
* create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Reliability
* Transmit channel
* Implement putRawString for some ugly char (_INIT packet), and use it.
* Many packet read and write migrated
* Implement oldForgePacket to interface writing with current connection
* fix U8/char/bool writing
* fix string writing and performances.
* add some missing functions
* Use v3s16 read instead of reading x,y,z separately
* Add irr::video::SColor support into packets
* Add some missing handlers
* Add a template function to increase offset
* Throw a serialization error on packet reading (must be improved)
PacketFactories:
* Create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Create ClientCommandFactory, used by server to get useful informations about packet processing (sending).
Client.cpp:
* implement NetworkPacket ::Send interface.
* Move packet handlers to a dedicated file
* Remove Client::Send(SharedBuffer)
Server.cpp:
* implement NetworkPacket ::Send interface.
* Rewrite all packets using NetworkPacket
* Move packet handlers to a dedicated file
* Remove Server::Send(SharedBuffer)
ClientIface.cpp:
* Remove sendToAll(SharedBuffer<u8>)
Connection.hpp rework:
* Remove duplicate include
* Remove duplicate negation
* Remove a useless variable
* Improve code performance by using a m_peers_list instead of scanning m_peers map
* Remove Connection::Send(SharedBuffer)
* Fix useafterfree into NetworkPacket Sending
* Remove unused Connection::sendToAll
Test.cpp:
* Remove dead code
* Update tests to use NetworkPackets
Misc:
* add new wrappers to Send packets in client, using NetworkPacket
* Add NetworkPacket methods for Connection
* coding style fix
* dead code since changes cleanup
* Use v3s16 read instead of reading x,y,z separately in some packets
* Use different files to handle packets received by client and server
* Cleanup: Remove useless includes
ok @Zeno-
Tested by @Zeno- @VanessaE and @nerzhul on running servers
2015-01-16 11:37:49 +01:00
|
|
|
<< animation_frames[3] << animation_speed;
|
2014-01-08 13:47:53 +01:00
|
|
|
|
2015-03-13 22:01:49 +01:00
|
|
|
Send(&pkt);
|
2014-01-08 13:47:53 +01:00
|
|
|
}
|
|
|
|
|
2017-09-27 19:47:36 +02:00
|
|
|
void Server::SendEyeOffset(session_t peer_id, v3f first, v3f third)
|
2014-04-11 15:32:46 +02:00
|
|
|
{
|
2015-03-13 22:01:49 +01:00
|
|
|
NetworkPacket pkt(TOCLIENT_EYE_OFFSET, 0, peer_id);
|
|
|
|
pkt << first << third;
|
|
|
|
Send(&pkt);
|
2014-04-11 15:32:46 +02:00
|
|
|
}
|
2017-09-27 19:47:36 +02:00
|
|
|
|
|
|
|
void Server::SendPlayerPrivileges(session_t peer_id)
|
2012-03-31 15:23:26 +02:00
|
|
|
{
|
2016-10-08 16:31:22 +02:00
|
|
|
RemotePlayer *player = m_env->getPlayer(peer_id);
|
2012-03-31 15:23:26 +02:00
|
|
|
assert(player);
|
2017-09-30 12:00:05 +02:00
|
|
|
if(player->getPeerId() == PEER_ID_INEXISTENT)
|
2012-07-19 13:09:16 +02:00
|
|
|
return;
|
|
|
|
|
2012-03-31 15:23:26 +02:00
|
|
|
std::set<std::string> privs;
|
2013-05-25 00:51:02 +02:00
|
|
|
m_script->getAuth(player->getName(), NULL, &privs);
|
2012-11-26 03:16:48 +01:00
|
|
|
|
2015-03-13 22:01:49 +01:00
|
|
|
NetworkPacket pkt(TOCLIENT_PRIVILEGES, 0, peer_id);
|
|
|
|
pkt << (u16) privs.size();
|
[Patch 2/4] Network rework: packet writing, sending and cleanups
NetworkPacket.cpp:
* Remove some deprecated functions, we must use streaming interface
* m_data converted from u8* to std::vector<u8>
* Add an exporter to forge packet to Connection object
* implement operator << std::wstring. n
* implement operator << std::string
* dynamic resize when write packet content.
* fix string writing and performances.
* create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Reliability
* Transmit channel
* Implement putRawString for some ugly char (_INIT packet), and use it.
* Many packet read and write migrated
* Implement oldForgePacket to interface writing with current connection
* fix U8/char/bool writing
* fix string writing and performances.
* add some missing functions
* Use v3s16 read instead of reading x,y,z separately
* Add irr::video::SColor support into packets
* Add some missing handlers
* Add a template function to increase offset
* Throw a serialization error on packet reading (must be improved)
PacketFactories:
* Create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Create ClientCommandFactory, used by server to get useful informations about packet processing (sending).
Client.cpp:
* implement NetworkPacket ::Send interface.
* Move packet handlers to a dedicated file
* Remove Client::Send(SharedBuffer)
Server.cpp:
* implement NetworkPacket ::Send interface.
* Rewrite all packets using NetworkPacket
* Move packet handlers to a dedicated file
* Remove Server::Send(SharedBuffer)
ClientIface.cpp:
* Remove sendToAll(SharedBuffer<u8>)
Connection.hpp rework:
* Remove duplicate include
* Remove duplicate negation
* Remove a useless variable
* Improve code performance by using a m_peers_list instead of scanning m_peers map
* Remove Connection::Send(SharedBuffer)
* Fix useafterfree into NetworkPacket Sending
* Remove unused Connection::sendToAll
Test.cpp:
* Remove dead code
* Update tests to use NetworkPackets
Misc:
* add new wrappers to Send packets in client, using NetworkPacket
* Add NetworkPacket methods for Connection
* coding style fix
* dead code since changes cleanup
* Use v3s16 read instead of reading x,y,z separately in some packets
* Use different files to handle packets received by client and server
* Cleanup: Remove useless includes
ok @Zeno-
Tested by @Zeno- @VanessaE and @nerzhul on running servers
2015-01-16 11:37:49 +01:00
|
|
|
|
2017-08-14 01:06:12 +02:00
|
|
|
for (const std::string &priv : privs) {
|
|
|
|
pkt << priv;
|
2012-03-31 15:23:26 +02:00
|
|
|
}
|
|
|
|
|
2015-03-13 22:01:49 +01:00
|
|
|
Send(&pkt);
|
2012-03-31 15:23:26 +02:00
|
|
|
}
|
|
|
|
|
2017-09-27 19:47:36 +02:00
|
|
|
void Server::SendPlayerInventoryFormspec(session_t peer_id)
|
2012-07-19 13:09:16 +02:00
|
|
|
{
|
2016-10-08 16:31:22 +02:00
|
|
|
RemotePlayer *player = m_env->getPlayer(peer_id);
|
2012-07-19 13:09:16 +02:00
|
|
|
assert(player);
|
2018-03-28 17:04:41 +02:00
|
|
|
if (player->getPeerId() == PEER_ID_INEXISTENT)
|
2012-07-19 13:09:16 +02:00
|
|
|
return;
|
|
|
|
|
2015-03-13 22:01:49 +01:00
|
|
|
NetworkPacket pkt(TOCLIENT_INVENTORY_FORMSPEC, 0, peer_id);
|
|
|
|
pkt.putLongString(FORMSPEC_VERSION_STRING + player->inventory_formspec);
|
|
|
|
Send(&pkt);
|
2012-07-19 13:09:16 +02:00
|
|
|
}
|
|
|
|
|
2018-03-28 17:04:41 +02:00
|
|
|
void Server::SendPlayerFormspecPrepend(session_t peer_id)
|
|
|
|
{
|
|
|
|
RemotePlayer *player = m_env->getPlayer(peer_id);
|
|
|
|
assert(player);
|
|
|
|
if (player->getPeerId() == PEER_ID_INEXISTENT)
|
|
|
|
return;
|
|
|
|
|
|
|
|
NetworkPacket pkt(TOCLIENT_FORMSPEC_PREPEND, 0, peer_id);
|
|
|
|
pkt << FORMSPEC_VERSION_STRING + player->formspec_prepend;
|
|
|
|
Send(&pkt);
|
|
|
|
}
|
|
|
|
|
2017-09-27 19:47:36 +02:00
|
|
|
u32 Server::SendActiveObjectRemoveAdd(session_t peer_id, const std::string &datas)
|
2015-03-16 12:01:40 +01:00
|
|
|
{
|
2015-04-05 11:37:53 +02:00
|
|
|
NetworkPacket pkt(TOCLIENT_ACTIVE_OBJECT_REMOVE_ADD, datas.size(), peer_id);
|
2015-03-16 12:01:40 +01:00
|
|
|
pkt.putRawString(datas.c_str(), datas.size());
|
|
|
|
Send(&pkt);
|
|
|
|
return pkt.getSize();
|
|
|
|
}
|
|
|
|
|
2017-09-27 19:47:36 +02:00
|
|
|
void Server::SendActiveObjectMessages(session_t peer_id, const std::string &datas,
|
|
|
|
bool reliable)
|
2015-03-16 12:01:40 +01:00
|
|
|
{
|
|
|
|
NetworkPacket pkt(TOCLIENT_ACTIVE_OBJECT_MESSAGES,
|
2015-04-05 11:37:53 +02:00
|
|
|
datas.size(), peer_id);
|
2015-03-16 12:01:40 +01:00
|
|
|
|
|
|
|
pkt.putRawString(datas.c_str(), datas.size());
|
2015-03-17 09:16:39 +01:00
|
|
|
|
|
|
|
m_clients.send(pkt.getPeerId(),
|
2015-04-07 14:28:36 +02:00
|
|
|
reliable ? clientCommandFactoryTable[pkt.getCommand()].channel : 1,
|
2015-03-17 09:16:39 +01:00
|
|
|
&pkt, reliable);
|
2017-07-18 21:39:55 +02:00
|
|
|
}
|
2015-03-17 09:16:39 +01:00
|
|
|
|
2018-06-20 22:36:08 +02:00
|
|
|
void Server::SendCSMRestrictionFlags(session_t peer_id)
|
2017-07-18 21:39:55 +02:00
|
|
|
{
|
2018-06-20 22:36:08 +02:00
|
|
|
NetworkPacket pkt(TOCLIENT_CSM_RESTRICTION_FLAGS,
|
|
|
|
sizeof(m_csm_restriction_flags) + sizeof(m_csm_restriction_noderange), peer_id);
|
|
|
|
pkt << m_csm_restriction_flags << m_csm_restriction_noderange;
|
2017-07-18 21:39:55 +02:00
|
|
|
Send(&pkt);
|
2015-03-16 12:01:40 +01:00
|
|
|
}
|
|
|
|
|
2012-03-24 18:01:26 +01:00
|
|
|
s32 Server::playSound(const SimpleSoundSpec &spec,
|
|
|
|
const ServerSoundParams ¶ms)
|
|
|
|
{
|
|
|
|
// Find out initial position of sound
|
|
|
|
bool pos_exists = false;
|
|
|
|
v3f pos = params.getPos(m_env, &pos_exists);
|
|
|
|
// If position is not found while it should be, cancel sound
|
|
|
|
if(pos_exists != (params.type != ServerSoundParams::SSP_LOCAL))
|
|
|
|
return -1;
|
2014-01-31 00:24:00 +01:00
|
|
|
|
2012-03-24 18:01:26 +01:00
|
|
|
// Filter destination clients
|
2017-09-27 23:48:06 +02:00
|
|
|
std::vector<session_t> dst_clients;
|
2017-08-14 01:06:12 +02:00
|
|
|
if(!params.to_player.empty()) {
|
2016-10-08 16:31:22 +02:00
|
|
|
RemotePlayer *player = m_env->getPlayer(params.to_player.c_str());
|
2012-03-24 18:01:26 +01:00
|
|
|
if(!player){
|
|
|
|
infostream<<"Server::playSound: Player \""<<params.to_player
|
|
|
|
<<"\" not found"<<std::endl;
|
|
|
|
return -1;
|
|
|
|
}
|
2017-09-30 12:00:05 +02:00
|
|
|
if (player->getPeerId() == PEER_ID_INEXISTENT) {
|
2012-03-24 18:01:26 +01:00
|
|
|
infostream<<"Server::playSound: Player \""<<params.to_player
|
|
|
|
<<"\" not connected"<<std::endl;
|
|
|
|
return -1;
|
|
|
|
}
|
2017-09-30 12:00:05 +02:00
|
|
|
dst_clients.push_back(player->getPeerId());
|
2017-08-14 01:06:12 +02:00
|
|
|
} else {
|
2017-09-27 23:48:06 +02:00
|
|
|
std::vector<session_t> clients = m_clients.getClientIDs();
|
2014-01-31 00:24:00 +01:00
|
|
|
|
2017-09-27 23:48:06 +02:00
|
|
|
for (const session_t client_id : clients) {
|
2017-08-14 01:06:12 +02:00
|
|
|
RemotePlayer *player = m_env->getPlayer(client_id);
|
2016-10-08 16:31:22 +02:00
|
|
|
if (!player)
|
2012-03-24 18:01:26 +01:00
|
|
|
continue;
|
2015-03-04 16:30:24 +01:00
|
|
|
|
2016-10-30 14:53:26 +01:00
|
|
|
PlayerSAO *sao = player->getPlayerSAO();
|
|
|
|
if (!sao)
|
|
|
|
continue;
|
|
|
|
|
2016-10-08 16:31:22 +02:00
|
|
|
if (pos_exists) {
|
2016-10-30 14:53:26 +01:00
|
|
|
if(sao->getBasePosition().getDistanceFrom(pos) >
|
2012-03-24 18:01:26 +01:00
|
|
|
params.max_hear_distance)
|
|
|
|
continue;
|
|
|
|
}
|
2017-08-14 01:06:12 +02:00
|
|
|
dst_clients.push_back(client_id);
|
2012-03-24 18:01:26 +01:00
|
|
|
}
|
|
|
|
}
|
2015-03-04 16:30:24 +01:00
|
|
|
|
2014-12-12 15:55:40 +01:00
|
|
|
if(dst_clients.empty())
|
2012-03-24 18:01:26 +01:00
|
|
|
return -1;
|
2014-01-31 00:24:00 +01:00
|
|
|
|
2012-03-24 18:01:26 +01:00
|
|
|
// Create the sound
|
|
|
|
s32 id = m_next_sound_id++;
|
|
|
|
// The sound will exist as a reference in m_playing_sounds
|
|
|
|
m_playing_sounds[id] = ServerPlayingSound();
|
|
|
|
ServerPlayingSound &psound = m_playing_sounds[id];
|
|
|
|
psound.params = params;
|
2016-07-10 07:08:26 +02:00
|
|
|
psound.spec = spec;
|
[Patch 2/4] Network rework: packet writing, sending and cleanups
NetworkPacket.cpp:
* Remove some deprecated functions, we must use streaming interface
* m_data converted from u8* to std::vector<u8>
* Add an exporter to forge packet to Connection object
* implement operator << std::wstring. n
* implement operator << std::string
* dynamic resize when write packet content.
* fix string writing and performances.
* create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Reliability
* Transmit channel
* Implement putRawString for some ugly char (_INIT packet), and use it.
* Many packet read and write migrated
* Implement oldForgePacket to interface writing with current connection
* fix U8/char/bool writing
* fix string writing and performances.
* add some missing functions
* Use v3s16 read instead of reading x,y,z separately
* Add irr::video::SColor support into packets
* Add some missing handlers
* Add a template function to increase offset
* Throw a serialization error on packet reading (must be improved)
PacketFactories:
* Create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Create ClientCommandFactory, used by server to get useful informations about packet processing (sending).
Client.cpp:
* implement NetworkPacket ::Send interface.
* Move packet handlers to a dedicated file
* Remove Client::Send(SharedBuffer)
Server.cpp:
* implement NetworkPacket ::Send interface.
* Rewrite all packets using NetworkPacket
* Move packet handlers to a dedicated file
* Remove Server::Send(SharedBuffer)
ClientIface.cpp:
* Remove sendToAll(SharedBuffer<u8>)
Connection.hpp rework:
* Remove duplicate include
* Remove duplicate negation
* Remove a useless variable
* Improve code performance by using a m_peers_list instead of scanning m_peers map
* Remove Connection::Send(SharedBuffer)
* Fix useafterfree into NetworkPacket Sending
* Remove unused Connection::sendToAll
Test.cpp:
* Remove dead code
* Update tests to use NetworkPackets
Misc:
* add new wrappers to Send packets in client, using NetworkPacket
* Add NetworkPacket methods for Connection
* coding style fix
* dead code since changes cleanup
* Use v3s16 read instead of reading x,y,z separately in some packets
* Use different files to handle packets received by client and server
* Cleanup: Remove useless includes
ok @Zeno-
Tested by @Zeno- @VanessaE and @nerzhul on running servers
2015-01-16 11:37:49 +01:00
|
|
|
|
2016-07-10 07:08:26 +02:00
|
|
|
float gain = params.gain * spec.gain;
|
2015-03-13 22:01:49 +01:00
|
|
|
NetworkPacket pkt(TOCLIENT_PLAY_SOUND, 0);
|
2016-07-10 07:08:26 +02:00
|
|
|
pkt << id << spec.name << gain
|
|
|
|
<< (u8) params.type << pos << params.object
|
2017-06-11 13:58:26 +02:00
|
|
|
<< params.loop << params.fade << params.pitch;
|
2015-03-05 11:39:05 +01:00
|
|
|
|
2016-07-10 07:08:26 +02:00
|
|
|
// Backwards compability
|
|
|
|
bool play_sound = gain > 0;
|
|
|
|
|
2017-08-14 01:06:12 +02:00
|
|
|
for (const u16 dst_client : dst_clients) {
|
|
|
|
if (play_sound || m_clients.getProtocolVersion(dst_client) >= 32) {
|
|
|
|
psound.clients.insert(dst_client);
|
|
|
|
m_clients.send(dst_client, 0, &pkt, true);
|
2016-07-10 07:08:26 +02:00
|
|
|
}
|
2012-03-24 18:01:26 +01:00
|
|
|
}
|
|
|
|
return id;
|
|
|
|
}
|
|
|
|
void Server::stopSound(s32 handle)
|
|
|
|
{
|
|
|
|
// Get sound reference
|
2017-06-04 21:00:04 +02:00
|
|
|
std::unordered_map<s32, ServerPlayingSound>::iterator i =
|
|
|
|
m_playing_sounds.find(handle);
|
2016-10-06 19:20:12 +02:00
|
|
|
if (i == m_playing_sounds.end())
|
2012-03-24 18:01:26 +01:00
|
|
|
return;
|
|
|
|
ServerPlayingSound &psound = i->second;
|
[Patch 2/4] Network rework: packet writing, sending and cleanups
NetworkPacket.cpp:
* Remove some deprecated functions, we must use streaming interface
* m_data converted from u8* to std::vector<u8>
* Add an exporter to forge packet to Connection object
* implement operator << std::wstring. n
* implement operator << std::string
* dynamic resize when write packet content.
* fix string writing and performances.
* create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Reliability
* Transmit channel
* Implement putRawString for some ugly char (_INIT packet), and use it.
* Many packet read and write migrated
* Implement oldForgePacket to interface writing with current connection
* fix U8/char/bool writing
* fix string writing and performances.
* add some missing functions
* Use v3s16 read instead of reading x,y,z separately
* Add irr::video::SColor support into packets
* Add some missing handlers
* Add a template function to increase offset
* Throw a serialization error on packet reading (must be improved)
PacketFactories:
* Create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Create ClientCommandFactory, used by server to get useful informations about packet processing (sending).
Client.cpp:
* implement NetworkPacket ::Send interface.
* Move packet handlers to a dedicated file
* Remove Client::Send(SharedBuffer)
Server.cpp:
* implement NetworkPacket ::Send interface.
* Rewrite all packets using NetworkPacket
* Move packet handlers to a dedicated file
* Remove Server::Send(SharedBuffer)
ClientIface.cpp:
* Remove sendToAll(SharedBuffer<u8>)
Connection.hpp rework:
* Remove duplicate include
* Remove duplicate negation
* Remove a useless variable
* Improve code performance by using a m_peers_list instead of scanning m_peers map
* Remove Connection::Send(SharedBuffer)
* Fix useafterfree into NetworkPacket Sending
* Remove unused Connection::sendToAll
Test.cpp:
* Remove dead code
* Update tests to use NetworkPackets
Misc:
* add new wrappers to Send packets in client, using NetworkPacket
* Add NetworkPacket methods for Connection
* coding style fix
* dead code since changes cleanup
* Use v3s16 read instead of reading x,y,z separately in some packets
* Use different files to handle packets received by client and server
* Cleanup: Remove useless includes
ok @Zeno-
Tested by @Zeno- @VanessaE and @nerzhul on running servers
2015-01-16 11:37:49 +01:00
|
|
|
|
2015-03-13 22:01:49 +01:00
|
|
|
NetworkPacket pkt(TOCLIENT_STOP_SOUND, 4);
|
|
|
|
pkt << handle;
|
[Patch 2/4] Network rework: packet writing, sending and cleanups
NetworkPacket.cpp:
* Remove some deprecated functions, we must use streaming interface
* m_data converted from u8* to std::vector<u8>
* Add an exporter to forge packet to Connection object
* implement operator << std::wstring. n
* implement operator << std::string
* dynamic resize when write packet content.
* fix string writing and performances.
* create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Reliability
* Transmit channel
* Implement putRawString for some ugly char (_INIT packet), and use it.
* Many packet read and write migrated
* Implement oldForgePacket to interface writing with current connection
* fix U8/char/bool writing
* fix string writing and performances.
* add some missing functions
* Use v3s16 read instead of reading x,y,z separately
* Add irr::video::SColor support into packets
* Add some missing handlers
* Add a template function to increase offset
* Throw a serialization error on packet reading (must be improved)
PacketFactories:
* Create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Create ClientCommandFactory, used by server to get useful informations about packet processing (sending).
Client.cpp:
* implement NetworkPacket ::Send interface.
* Move packet handlers to a dedicated file
* Remove Client::Send(SharedBuffer)
Server.cpp:
* implement NetworkPacket ::Send interface.
* Rewrite all packets using NetworkPacket
* Move packet handlers to a dedicated file
* Remove Server::Send(SharedBuffer)
ClientIface.cpp:
* Remove sendToAll(SharedBuffer<u8>)
Connection.hpp rework:
* Remove duplicate include
* Remove duplicate negation
* Remove a useless variable
* Improve code performance by using a m_peers_list instead of scanning m_peers map
* Remove Connection::Send(SharedBuffer)
* Fix useafterfree into NetworkPacket Sending
* Remove unused Connection::sendToAll
Test.cpp:
* Remove dead code
* Update tests to use NetworkPackets
Misc:
* add new wrappers to Send packets in client, using NetworkPacket
* Add NetworkPacket methods for Connection
* coding style fix
* dead code since changes cleanup
* Use v3s16 read instead of reading x,y,z separately in some packets
* Use different files to handle packets received by client and server
* Cleanup: Remove useless includes
ok @Zeno-
Tested by @Zeno- @VanessaE and @nerzhul on running servers
2015-01-16 11:37:49 +01:00
|
|
|
|
2017-09-27 23:48:06 +02:00
|
|
|
for (std::unordered_set<session_t>::const_iterator si = psound.clients.begin();
|
2017-06-04 21:00:04 +02:00
|
|
|
si != psound.clients.end(); ++si) {
|
2012-03-24 18:01:26 +01:00
|
|
|
// Send as reliable
|
2017-06-04 21:00:04 +02:00
|
|
|
m_clients.send(*si, 0, &pkt, true);
|
2012-03-24 18:01:26 +01:00
|
|
|
}
|
|
|
|
// Remove sound reference
|
|
|
|
m_playing_sounds.erase(i);
|
|
|
|
}
|
|
|
|
|
2016-07-10 07:08:26 +02:00
|
|
|
void Server::fadeSound(s32 handle, float step, float gain)
|
|
|
|
{
|
|
|
|
// Get sound reference
|
2017-06-04 21:00:04 +02:00
|
|
|
std::unordered_map<s32, ServerPlayingSound>::iterator i =
|
|
|
|
m_playing_sounds.find(handle);
|
2016-07-10 07:08:26 +02:00
|
|
|
if (i == m_playing_sounds.end())
|
|
|
|
return;
|
|
|
|
|
|
|
|
ServerPlayingSound &psound = i->second;
|
|
|
|
psound.params.gain = gain;
|
|
|
|
|
|
|
|
NetworkPacket pkt(TOCLIENT_FADE_SOUND, 4);
|
|
|
|
pkt << handle << step << gain;
|
|
|
|
|
|
|
|
// Backwards compability
|
|
|
|
bool play_sound = gain > 0;
|
|
|
|
ServerPlayingSound compat_psound = psound;
|
|
|
|
compat_psound.clients.clear();
|
|
|
|
|
|
|
|
NetworkPacket compat_pkt(TOCLIENT_STOP_SOUND, 4);
|
|
|
|
compat_pkt << handle;
|
|
|
|
|
2017-06-04 21:00:04 +02:00
|
|
|
for (std::unordered_set<u16>::iterator it = psound.clients.begin();
|
2016-07-10 07:08:26 +02:00
|
|
|
it != psound.clients.end();) {
|
|
|
|
if (m_clients.getProtocolVersion(*it) >= 32) {
|
|
|
|
// Send as reliable
|
|
|
|
m_clients.send(*it, 0, &pkt, true);
|
|
|
|
++it;
|
|
|
|
} else {
|
|
|
|
compat_psound.clients.insert(*it);
|
|
|
|
// Stop old sound
|
|
|
|
m_clients.send(*it, 0, &compat_pkt, true);
|
|
|
|
psound.clients.erase(it++);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Remove sound reference
|
2017-08-14 01:06:12 +02:00
|
|
|
if (!play_sound || psound.clients.empty())
|
2016-07-10 07:08:26 +02:00
|
|
|
m_playing_sounds.erase(i);
|
|
|
|
|
2017-08-14 01:06:12 +02:00
|
|
|
if (play_sound && !compat_psound.clients.empty()) {
|
2016-07-10 07:08:26 +02:00
|
|
|
// Play new sound volume on older clients
|
|
|
|
playSound(compat_psound.spec, compat_psound.params);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-08-16 20:10:34 +02:00
|
|
|
void Server::sendRemoveNode(v3s16 p, std::unordered_set<u16> *far_players,
|
|
|
|
float far_d_nodes)
|
2011-04-21 18:35:17 +02:00
|
|
|
{
|
2018-08-16 20:10:34 +02:00
|
|
|
float maxd = far_d_nodes * BS;
|
2011-04-21 18:35:17 +02:00
|
|
|
v3f p_f = intToFloat(p, BS);
|
2018-08-16 20:10:34 +02:00
|
|
|
v3s16 block_pos = getNodeBlockPos(p);
|
2011-04-21 18:35:17 +02:00
|
|
|
|
2015-03-13 22:01:49 +01:00
|
|
|
NetworkPacket pkt(TOCLIENT_REMOVENODE, 6);
|
|
|
|
pkt << p;
|
2011-04-21 18:35:17 +02:00
|
|
|
|
2017-09-27 23:48:06 +02:00
|
|
|
std::vector<session_t> clients = m_clients.getClientIDs();
|
2018-08-16 20:10:34 +02:00
|
|
|
m_clients.lock();
|
2016-10-30 14:53:26 +01:00
|
|
|
|
2018-08-16 20:10:34 +02:00
|
|
|
for (session_t client_id : clients) {
|
|
|
|
RemoteClient *client = m_clients.lockedGetClientNoEx(client_id);
|
|
|
|
if (!client)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
RemotePlayer *player = m_env->getPlayer(client_id);
|
|
|
|
PlayerSAO *sao = player ? player->getPlayerSAO() : nullptr;
|
|
|
|
|
|
|
|
// If player is far away, only set modified blocks not sent
|
|
|
|
if (!client->isBlockSent(block_pos) || (sao &&
|
|
|
|
sao->getBasePosition().getDistanceFrom(p_f) > maxd)) {
|
|
|
|
if (far_players)
|
|
|
|
far_players->emplace(client_id);
|
|
|
|
else
|
|
|
|
client->SetBlockNotSent(block_pos);
|
|
|
|
continue;
|
2011-04-21 18:35:17 +02:00
|
|
|
}
|
2010-12-25 00:54:39 +01:00
|
|
|
|
2011-04-21 18:35:17 +02:00
|
|
|
// Send as reliable
|
2017-08-15 09:30:31 +02:00
|
|
|
m_clients.send(client_id, 0, &pkt, true);
|
2011-04-21 18:35:17 +02:00
|
|
|
}
|
2018-08-16 20:10:34 +02:00
|
|
|
|
|
|
|
m_clients.unlock();
|
2011-04-21 18:35:17 +02:00
|
|
|
}
|
|
|
|
|
2018-08-16 20:10:34 +02:00
|
|
|
void Server::sendAddNode(v3s16 p, MapNode n, std::unordered_set<u16> *far_players,
|
|
|
|
float far_d_nodes, bool remove_metadata)
|
2011-04-21 18:35:17 +02:00
|
|
|
{
|
2018-08-16 20:10:34 +02:00
|
|
|
float maxd = far_d_nodes * BS;
|
2011-04-21 18:35:17 +02:00
|
|
|
v3f p_f = intToFloat(p, BS);
|
2018-08-16 20:10:34 +02:00
|
|
|
v3s16 block_pos = getNodeBlockPos(p);
|
2011-04-21 18:35:17 +02:00
|
|
|
|
2018-08-16 20:10:34 +02:00
|
|
|
NetworkPacket pkt(TOCLIENT_ADDNODE, 6 + 2 + 1 + 1 + 1);
|
|
|
|
pkt << p << n.param0 << n.param1 << n.param2
|
|
|
|
<< (u8) (remove_metadata ? 0 : 1);
|
2016-10-30 14:53:26 +01:00
|
|
|
|
2018-08-16 20:10:34 +02:00
|
|
|
std::vector<session_t> clients = m_clients.getClientIDs();
|
|
|
|
m_clients.lock();
|
[Patch 2/4] Network rework: packet writing, sending and cleanups
NetworkPacket.cpp:
* Remove some deprecated functions, we must use streaming interface
* m_data converted from u8* to std::vector<u8>
* Add an exporter to forge packet to Connection object
* implement operator << std::wstring. n
* implement operator << std::string
* dynamic resize when write packet content.
* fix string writing and performances.
* create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Reliability
* Transmit channel
* Implement putRawString for some ugly char (_INIT packet), and use it.
* Many packet read and write migrated
* Implement oldForgePacket to interface writing with current connection
* fix U8/char/bool writing
* fix string writing and performances.
* add some missing functions
* Use v3s16 read instead of reading x,y,z separately
* Add irr::video::SColor support into packets
* Add some missing handlers
* Add a template function to increase offset
* Throw a serialization error on packet reading (must be improved)
PacketFactories:
* Create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Create ClientCommandFactory, used by server to get useful informations about packet processing (sending).
Client.cpp:
* implement NetworkPacket ::Send interface.
* Move packet handlers to a dedicated file
* Remove Client::Send(SharedBuffer)
Server.cpp:
* implement NetworkPacket ::Send interface.
* Rewrite all packets using NetworkPacket
* Move packet handlers to a dedicated file
* Remove Server::Send(SharedBuffer)
ClientIface.cpp:
* Remove sendToAll(SharedBuffer<u8>)
Connection.hpp rework:
* Remove duplicate include
* Remove duplicate negation
* Remove a useless variable
* Improve code performance by using a m_peers_list instead of scanning m_peers map
* Remove Connection::Send(SharedBuffer)
* Fix useafterfree into NetworkPacket Sending
* Remove unused Connection::sendToAll
Test.cpp:
* Remove dead code
* Update tests to use NetworkPackets
Misc:
* add new wrappers to Send packets in client, using NetworkPacket
* Add NetworkPacket methods for Connection
* coding style fix
* dead code since changes cleanup
* Use v3s16 read instead of reading x,y,z separately in some packets
* Use different files to handle packets received by client and server
* Cleanup: Remove useless includes
ok @Zeno-
Tested by @Zeno- @VanessaE and @nerzhul on running servers
2015-01-16 11:37:49 +01:00
|
|
|
|
2018-08-16 20:10:34 +02:00
|
|
|
for (session_t client_id : clients) {
|
|
|
|
RemoteClient *client = m_clients.lockedGetClientNoEx(client_id);
|
|
|
|
if (!client)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
RemotePlayer *player = m_env->getPlayer(client_id);
|
|
|
|
PlayerSAO *sao = player ? player->getPlayerSAO() : nullptr;
|
|
|
|
|
|
|
|
// If player is far away, only set modified blocks not sent
|
|
|
|
if (!client->isBlockSent(block_pos) || (sao &&
|
|
|
|
sao->getBasePosition().getDistanceFrom(p_f) > maxd)) {
|
|
|
|
if (far_players)
|
|
|
|
far_players->emplace(client_id);
|
|
|
|
else
|
|
|
|
client->SetBlockNotSent(block_pos);
|
|
|
|
continue;
|
2013-11-23 15:35:49 +01:00
|
|
|
}
|
2011-04-21 18:35:17 +02:00
|
|
|
|
|
|
|
// Send as reliable
|
2018-08-16 20:10:34 +02:00
|
|
|
m_clients.send(client_id, 0, &pkt, true);
|
2011-04-21 18:35:17 +02:00
|
|
|
}
|
2018-08-16 20:10:34 +02:00
|
|
|
|
|
|
|
m_clients.unlock();
|
2011-04-21 18:35:17 +02:00
|
|
|
}
|
|
|
|
|
2018-12-04 20:37:48 +01:00
|
|
|
void Server::sendMetadataChanged(const std::list<v3s16> &meta_updates, float far_d_nodes)
|
|
|
|
{
|
|
|
|
float maxd = far_d_nodes * BS;
|
|
|
|
NodeMetadataList meta_updates_list(false);
|
|
|
|
std::vector<session_t> clients = m_clients.getClientIDs();
|
|
|
|
|
|
|
|
m_clients.lock();
|
|
|
|
|
|
|
|
for (session_t i : clients) {
|
|
|
|
RemoteClient *client = m_clients.lockedGetClientNoEx(i);
|
|
|
|
if (!client)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
ServerActiveObject *player = m_env->getActiveObject(i);
|
|
|
|
v3f player_pos = player ? player->getBasePosition() : v3f();
|
|
|
|
|
|
|
|
for (const v3s16 &pos : meta_updates) {
|
|
|
|
NodeMetadata *meta = m_env->getMap().getNodeMetadata(pos);
|
|
|
|
|
|
|
|
if (!meta)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
v3s16 block_pos = getNodeBlockPos(pos);
|
|
|
|
if (!client->isBlockSent(block_pos) || (player &&
|
|
|
|
player_pos.getDistanceFrom(intToFloat(pos, BS)) > maxd)) {
|
|
|
|
client->SetBlockNotSent(block_pos);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add the change to send list
|
|
|
|
meta_updates_list.set(pos, meta);
|
|
|
|
}
|
|
|
|
if (meta_updates_list.size() == 0)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
// Send the meta changes
|
|
|
|
std::ostringstream os(std::ios::binary);
|
|
|
|
meta_updates_list.serialize(os, client->net_proto_version, false, true);
|
|
|
|
std::ostringstream oss(std::ios::binary);
|
|
|
|
compressZlib(os.str(), oss);
|
|
|
|
|
|
|
|
NetworkPacket pkt(TOCLIENT_NODEMETA_CHANGED, 0);
|
|
|
|
pkt.putLongString(oss.str());
|
|
|
|
m_clients.send(i, 0, &pkt, true);
|
|
|
|
|
|
|
|
meta_updates_list.clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
m_clients.unlock();
|
|
|
|
}
|
|
|
|
|
2017-09-27 19:47:36 +02:00
|
|
|
void Server::SendBlockNoLock(session_t peer_id, MapBlock *block, u8 ver,
|
|
|
|
u16 net_proto_version)
|
2011-04-21 18:35:17 +02:00
|
|
|
{
|
|
|
|
/*
|
|
|
|
Create a packet with the block in the right format
|
|
|
|
*/
|
2012-11-26 03:16:48 +01:00
|
|
|
|
2011-04-21 18:35:17 +02:00
|
|
|
std::ostringstream os(std::ios_base::binary);
|
2012-01-21 00:11:44 +01:00
|
|
|
block->serialize(os, ver, false);
|
2017-03-19 08:44:29 +01:00
|
|
|
block->serializeNetworkSpecific(os);
|
2011-04-21 18:35:17 +02:00
|
|
|
std::string s = os.str();
|
|
|
|
|
2015-03-13 22:01:49 +01:00
|
|
|
NetworkPacket pkt(TOCLIENT_BLOCKDATA, 2 + 2 + 2 + 2 + s.size(), peer_id);
|
2011-04-21 18:35:17 +02:00
|
|
|
|
2018-03-08 22:58:43 +01:00
|
|
|
pkt << block->getPos();
|
2015-03-13 22:01:49 +01:00
|
|
|
pkt.putRawString(s.c_str(), s.size());
|
|
|
|
Send(&pkt);
|
2011-04-21 18:35:17 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void Server::SendBlocks(float dtime)
|
|
|
|
{
|
2015-04-07 12:13:12 +02:00
|
|
|
MutexAutoLock envlock(m_env_mutex);
|
2014-01-31 00:24:00 +01:00
|
|
|
//TODO check if one big lock could be faster then multiple small ones
|
2011-04-21 18:35:17 +02:00
|
|
|
|
2012-03-07 20:54:18 +01:00
|
|
|
ScopeProfiler sp(g_profiler, "Server: sel and send blocks to clients");
|
2011-04-21 18:35:17 +02:00
|
|
|
|
2012-12-20 18:19:49 +01:00
|
|
|
std::vector<PrioritySortedBlockTransfer> queue;
|
2011-04-21 18:35:17 +02:00
|
|
|
|
2017-09-09 00:36:48 +02:00
|
|
|
u32 total_sending = 0;
|
2012-11-26 03:16:48 +01:00
|
|
|
|
2011-04-21 18:35:17 +02:00
|
|
|
{
|
2017-08-14 00:44:45 +02:00
|
|
|
ScopeProfiler sp2(g_profiler, "Server: selecting blocks for sending");
|
2011-04-21 18:35:17 +02:00
|
|
|
|
2017-09-27 23:48:06 +02:00
|
|
|
std::vector<session_t> clients = m_clients.getClientIDs();
|
2014-01-31 00:24:00 +01:00
|
|
|
|
2015-04-07 12:13:12 +02:00
|
|
|
m_clients.lock();
|
2017-09-27 23:48:06 +02:00
|
|
|
for (const session_t client_id : clients) {
|
2017-08-14 00:44:45 +02:00
|
|
|
RemoteClient *client = m_clients.lockedGetClientNoEx(client_id, CS_Active);
|
2011-05-30 23:15:43 +02:00
|
|
|
|
2017-08-14 00:44:45 +02:00
|
|
|
if (!client)
|
2014-09-12 02:12:52 +02:00
|
|
|
continue;
|
2011-12-02 00:18:25 +01:00
|
|
|
|
2017-09-09 00:36:48 +02:00
|
|
|
total_sending += client->getSendingCount();
|
2014-01-31 00:24:00 +01:00
|
|
|
client->GetNextBlocks(m_env,m_emerge, dtime, queue);
|
2011-05-30 23:15:43 +02:00
|
|
|
}
|
2015-04-07 12:13:12 +02:00
|
|
|
m_clients.unlock();
|
2011-04-21 18:35:17 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Sort.
|
|
|
|
// Lowest priority number comes first.
|
|
|
|
// Lowest is most important.
|
2012-12-20 18:19:49 +01:00
|
|
|
std::sort(queue.begin(), queue.end());
|
2011-04-21 18:35:17 +02:00
|
|
|
|
2015-04-07 12:13:12 +02:00
|
|
|
m_clients.lock();
|
2017-09-09 00:36:48 +02:00
|
|
|
|
|
|
|
// Maximal total count calculation
|
|
|
|
// The per-client block sends is halved with the maximal online users
|
|
|
|
u32 max_blocks_to_send = (m_env->getPlayerCount() + g_settings->getU32("max_users")) *
|
|
|
|
g_settings->getU32("max_simultaneous_block_sends_per_client") / 4 + 1;
|
2017-08-18 12:17:30 +02:00
|
|
|
|
|
|
|
for (const PrioritySortedBlockTransfer &block_to_send : queue) {
|
|
|
|
if (total_sending >= max_blocks_to_send)
|
2011-04-21 18:35:17 +02:00
|
|
|
break;
|
2012-11-26 03:16:48 +01:00
|
|
|
|
2017-08-15 09:30:31 +02:00
|
|
|
MapBlock *block = nullptr;
|
|
|
|
try {
|
2017-08-18 12:17:30 +02:00
|
|
|
block = m_env->getMap().getBlockNoCreate(block_to_send.pos);
|
|
|
|
} catch (const InvalidPositionException &e) {
|
2011-04-21 18:35:17 +02:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2017-08-18 12:17:30 +02:00
|
|
|
RemoteClient *client = m_clients.lockedGetClientNoEx(block_to_send.peer_id,
|
|
|
|
CS_Active);
|
2017-08-15 09:30:31 +02:00
|
|
|
if (!client)
|
2013-08-04 07:17:07 +02:00
|
|
|
continue;
|
2011-04-21 18:35:17 +02:00
|
|
|
|
2017-08-18 12:17:30 +02:00
|
|
|
SendBlockNoLock(block_to_send.peer_id, block, client->serialization_version,
|
|
|
|
client->net_proto_version);
|
2011-04-21 18:35:17 +02:00
|
|
|
|
2017-08-18 12:17:30 +02:00
|
|
|
client->SentBlock(block_to_send.pos);
|
2011-04-21 18:35:17 +02:00
|
|
|
total_sending++;
|
|
|
|
}
|
2015-04-07 12:13:12 +02:00
|
|
|
m_clients.unlock();
|
2011-04-21 18:35:17 +02:00
|
|
|
}
|
|
|
|
|
2012-03-25 10:50:29 +02:00
|
|
|
void Server::fillMediaCache()
|
2012-03-11 03:15:45 +01:00
|
|
|
{
|
2012-03-25 11:10:58 +02:00
|
|
|
infostream<<"Server: Calculating media file checksums"<<std::endl;
|
2012-11-26 03:16:48 +01:00
|
|
|
|
2012-03-25 11:10:58 +02:00
|
|
|
// Collect all media file paths
|
2015-03-05 11:39:05 +01:00
|
|
|
std::vector<std::string> paths;
|
2018-03-16 08:41:33 +01:00
|
|
|
m_modmgr->getModsMediaPaths(paths);
|
2017-06-23 02:51:57 +02:00
|
|
|
fs::GetRecursiveDirs(paths, m_gamespec.path + DIR_DELIM + "textures");
|
|
|
|
fs::GetRecursiveDirs(paths, porting::path_user + DIR_DELIM + "textures" + DIR_DELIM + "server");
|
|
|
|
|
2012-03-25 11:10:58 +02:00
|
|
|
// Collect media file information from paths into cache
|
2017-08-19 14:25:35 +02:00
|
|
|
for (const std::string &mediapath : paths) {
|
2012-03-25 11:10:58 +02:00
|
|
|
std::vector<fs::DirListNode> dirlist = fs::GetDirListing(mediapath);
|
2017-08-19 14:25:35 +02:00
|
|
|
for (const fs::DirListNode &dln : dirlist) {
|
|
|
|
if (dln.dir) // Ignode dirs
|
2012-03-11 03:15:45 +01:00
|
|
|
continue;
|
2017-08-19 14:25:35 +02:00
|
|
|
std::string filename = dln.name;
|
2012-03-30 21:49:44 +02:00
|
|
|
// If name contains illegal characters, ignore the file
|
2015-03-05 11:39:05 +01:00
|
|
|
if (!string_allowed(filename, TEXTURENAME_ALLOWED_CHARS)) {
|
2012-03-30 21:49:44 +02:00
|
|
|
infostream<<"Server: ignoring illegal file name: \""
|
2015-03-05 11:39:05 +01:00
|
|
|
<< filename << "\"" << std::endl;
|
2012-03-11 03:15:45 +01:00
|
|
|
continue;
|
|
|
|
}
|
2012-03-30 21:49:44 +02:00
|
|
|
// If name is not in a supported format, ignore it
|
|
|
|
const char *supported_ext[] = {
|
|
|
|
".png", ".jpg", ".bmp", ".tga",
|
|
|
|
".pcx", ".ppm", ".psd", ".wal", ".rgb",
|
|
|
|
".ogg",
|
2012-10-24 21:10:05 +02:00
|
|
|
".x", ".b3d", ".md2", ".obj",
|
2017-01-31 18:05:03 +01:00
|
|
|
// Custom translation file format
|
|
|
|
".tr",
|
2012-03-30 21:49:44 +02:00
|
|
|
NULL
|
|
|
|
};
|
2017-08-14 01:06:12 +02:00
|
|
|
if (removeStringEnd(filename, supported_ext).empty()){
|
2015-03-05 11:39:05 +01:00
|
|
|
infostream << "Server: ignoring unsupported file extension: \""
|
|
|
|
<< filename << "\"" << std::endl;
|
2012-03-30 21:49:44 +02:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
// Ok, attempt to load the file and add to cache
|
2017-08-19 14:25:35 +02:00
|
|
|
std::string filepath;
|
|
|
|
filepath.append(mediapath).append(DIR_DELIM).append(filename);
|
|
|
|
|
2012-03-11 03:15:45 +01:00
|
|
|
// Read data
|
2012-03-25 11:10:58 +02:00
|
|
|
std::ifstream fis(filepath.c_str(), std::ios_base::binary);
|
2015-03-05 11:39:05 +01:00
|
|
|
if (!fis.good()) {
|
|
|
|
errorstream << "Server::fillMediaCache(): Could not open \""
|
|
|
|
<< filename << "\" for reading" << std::endl;
|
2012-03-11 03:15:45 +01:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
std::ostringstream tmp_os(std::ios_base::binary);
|
|
|
|
bool bad = false;
|
2015-03-05 11:39:05 +01:00
|
|
|
for(;;) {
|
2012-03-11 03:15:45 +01:00
|
|
|
char buf[1024];
|
|
|
|
fis.read(buf, 1024);
|
|
|
|
std::streamsize len = fis.gcount();
|
|
|
|
tmp_os.write(buf, len);
|
2015-03-05 11:39:05 +01:00
|
|
|
if (fis.eof())
|
2012-03-11 03:15:45 +01:00
|
|
|
break;
|
2015-03-05 11:39:05 +01:00
|
|
|
if (!fis.good()) {
|
2012-03-11 03:15:45 +01:00
|
|
|
bad = true;
|
|
|
|
break;
|
2012-01-03 23:37:46 +01:00
|
|
|
}
|
2012-03-11 03:15:45 +01:00
|
|
|
}
|
2015-03-05 11:39:05 +01:00
|
|
|
if(bad) {
|
2012-03-25 10:50:29 +02:00
|
|
|
errorstream<<"Server::fillMediaCache(): Failed to read \""
|
2015-03-05 11:39:05 +01:00
|
|
|
<< filename << "\"" << std::endl;
|
2012-03-11 03:15:45 +01:00
|
|
|
continue;
|
|
|
|
}
|
2015-03-05 11:39:05 +01:00
|
|
|
if(tmp_os.str().length() == 0) {
|
|
|
|
errorstream << "Server::fillMediaCache(): Empty file \""
|
|
|
|
<< filepath << "\"" << std::endl;
|
2012-03-11 03:15:45 +01:00
|
|
|
continue;
|
|
|
|
}
|
2012-01-02 12:31:50 +01:00
|
|
|
|
2012-03-11 03:15:45 +01:00
|
|
|
SHA1 sha1;
|
|
|
|
sha1.addBytes(tmp_os.str().c_str(), tmp_os.str().length());
|
2012-01-02 12:31:50 +01:00
|
|
|
|
2012-03-11 03:15:45 +01:00
|
|
|
unsigned char *digest = sha1.getDigest();
|
2012-03-25 13:47:51 +02:00
|
|
|
std::string sha1_base64 = base64_encode(digest, 20);
|
|
|
|
std::string sha1_hex = hex_encode((char*)digest, 20);
|
2012-03-11 03:15:45 +01:00
|
|
|
free(digest);
|
2012-01-02 12:31:50 +01:00
|
|
|
|
2012-03-11 03:15:45 +01:00
|
|
|
// Put in list
|
2015-03-05 11:39:05 +01:00
|
|
|
m_media[filename] = MediaInfo(filepath, sha1_base64);
|
|
|
|
verbosestream << "Server: " << sha1_hex << " is " << filename
|
|
|
|
<< std::endl;
|
2012-03-11 03:15:45 +01:00
|
|
|
}
|
2012-01-02 12:31:50 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-09-27 19:47:36 +02:00
|
|
|
void Server::sendMediaAnnouncement(session_t peer_id, const std::string &lang_code)
|
2012-03-25 10:50:29 +02:00
|
|
|
{
|
2015-07-30 23:14:05 +02:00
|
|
|
verbosestream << "Server: Announcing files to id(" << peer_id << ")"
|
|
|
|
<< std::endl;
|
2012-01-02 12:31:50 +01:00
|
|
|
|
2012-03-25 10:50:29 +02:00
|
|
|
// Make packet
|
2015-03-13 22:01:49 +01:00
|
|
|
NetworkPacket pkt(TOCLIENT_ANNOUNCE_MEDIA, 0, peer_id);
|
2017-01-31 18:05:03 +01:00
|
|
|
|
|
|
|
u16 media_sent = 0;
|
|
|
|
std::string lang_suffix;
|
|
|
|
lang_suffix.append(".").append(lang_code).append(".tr");
|
|
|
|
for (const auto &i : m_media) {
|
|
|
|
if (str_ends_with(i.first, ".tr") && !str_ends_with(i.first, lang_suffix))
|
|
|
|
continue;
|
|
|
|
media_sent++;
|
|
|
|
}
|
|
|
|
|
|
|
|
pkt << media_sent;
|
2012-01-02 12:31:50 +01:00
|
|
|
|
2017-08-14 01:06:12 +02:00
|
|
|
for (const auto &i : m_media) {
|
2017-01-31 18:05:03 +01:00
|
|
|
if (str_ends_with(i.first, ".tr") && !str_ends_with(i.first, lang_suffix))
|
|
|
|
continue;
|
2017-08-14 01:06:12 +02:00
|
|
|
pkt << i.first << i.second.sha1_digest;
|
2012-01-02 12:31:50 +01:00
|
|
|
}
|
|
|
|
|
2015-03-13 22:01:49 +01:00
|
|
|
pkt << g_settings->get("remote_media");
|
|
|
|
Send(&pkt);
|
2012-01-02 12:31:50 +01:00
|
|
|
}
|
|
|
|
|
2012-03-25 10:50:29 +02:00
|
|
|
struct SendableMedia
|
2011-11-15 10:02:47 +01:00
|
|
|
{
|
|
|
|
std::string name;
|
|
|
|
std::string path;
|
|
|
|
std::string data;
|
|
|
|
|
2014-02-27 21:12:59 +01:00
|
|
|
SendableMedia(const std::string &name_="", const std::string &path_="",
|
|
|
|
const std::string &data_=""):
|
2011-11-15 10:02:47 +01:00
|
|
|
name(name_),
|
|
|
|
path(path_),
|
|
|
|
data(data_)
|
|
|
|
{}
|
|
|
|
};
|
|
|
|
|
2017-09-27 19:47:36 +02:00
|
|
|
void Server::sendRequestedMedia(session_t peer_id,
|
2015-03-05 11:39:05 +01:00
|
|
|
const std::vector<std::string> &tosend)
|
2012-03-25 10:50:29 +02:00
|
|
|
{
|
|
|
|
verbosestream<<"Server::sendRequestedMedia(): "
|
|
|
|
<<"Sending files to client"<<std::endl;
|
2012-01-02 12:31:50 +01:00
|
|
|
|
2012-03-25 10:50:29 +02:00
|
|
|
/* Read files */
|
2012-01-02 12:31:50 +01:00
|
|
|
|
2011-11-15 22:58:56 +01:00
|
|
|
// Put 5kB in one bunch (this is not accurate)
|
|
|
|
u32 bytes_per_bunch = 5000;
|
2012-01-02 12:31:50 +01:00
|
|
|
|
2015-03-05 11:39:05 +01:00
|
|
|
std::vector< std::vector<SendableMedia> > file_bunches;
|
2017-08-14 01:06:12 +02:00
|
|
|
file_bunches.emplace_back();
|
2012-01-02 12:31:50 +01:00
|
|
|
|
2012-03-25 10:50:29 +02:00
|
|
|
u32 file_size_bunch_total = 0;
|
2012-01-02 12:31:50 +01:00
|
|
|
|
2017-08-14 01:06:12 +02:00
|
|
|
for (const std::string &name : tosend) {
|
2016-10-06 19:20:12 +02:00
|
|
|
if (m_media.find(name) == m_media.end()) {
|
2012-03-25 10:50:29 +02:00
|
|
|
errorstream<<"Server::sendRequestedMedia(): Client asked for "
|
Rewrite client media download and support hash-based remote download
Move most of the media-related code in client.cpp into a new class
ClientMediaDownloader (clientmedia.cpp, clientmedia.h). Among other
things, this class does the following things:
- Download [remote_server][sha1] instead of [remote_server][name]. This
is to support servers that provide the same file name with different
contents.
- Initially fetch [remote_server]index.mth. This file should follow the
Minetest Hashset format (currently version 1) and contain a list of SHA1
hashes that exist on the server.
- The list of needed SHA1s is uploaded (via HTTP POST) when index.mth is
requested, so servers can optionally narrow down the list to the needs
of the client.
- If index.mth is missing (HTTP response code 404), we enter compat mode,
fetching [remote_server][name] as before this commit.
- remote_server can now contain multiple servers, separated by commas.
The downloader code attempts to split requests between the different
servers, as permitted by each server's index.mth. If one server claims
to have a file but actually doesn't (or something fails), we ask a
different server that also claims to have it.
- As before, when none of the remote servers provide a particular
file, we download it via the conventional method, i.e. using
the minetest protocol: TOSERVER_REQUEST_MEDIA / TOCLIENT_MEDIA.
- Bugfix: Every downloaded file's SHA1 is now verified against the SHA1
announced by the minetest server (before loading it and inserting it
into the file cache).
- Bugfix: Only send TOSERVER_RECEIVED_MEDIA when we actually have all
media. This should fix #863.
2013-08-29 05:22:18 +02:00
|
|
|
<<"unknown file \""<<(name)<<"\""<<std::endl;
|
2012-01-03 23:37:46 +01:00
|
|
|
continue;
|
|
|
|
}
|
2012-01-02 12:31:50 +01:00
|
|
|
|
|
|
|
//TODO get path + name
|
Rewrite client media download and support hash-based remote download
Move most of the media-related code in client.cpp into a new class
ClientMediaDownloader (clientmedia.cpp, clientmedia.h). Among other
things, this class does the following things:
- Download [remote_server][sha1] instead of [remote_server][name]. This
is to support servers that provide the same file name with different
contents.
- Initially fetch [remote_server]index.mth. This file should follow the
Minetest Hashset format (currently version 1) and contain a list of SHA1
hashes that exist on the server.
- The list of needed SHA1s is uploaded (via HTTP POST) when index.mth is
requested, so servers can optionally narrow down the list to the needs
of the client.
- If index.mth is missing (HTTP response code 404), we enter compat mode,
fetching [remote_server][name] as before this commit.
- remote_server can now contain multiple servers, separated by commas.
The downloader code attempts to split requests between the different
servers, as permitted by each server's index.mth. If one server claims
to have a file but actually doesn't (or something fails), we ask a
different server that also claims to have it.
- As before, when none of the remote servers provide a particular
file, we download it via the conventional method, i.e. using
the minetest protocol: TOSERVER_REQUEST_MEDIA / TOCLIENT_MEDIA.
- Bugfix: Every downloaded file's SHA1 is now verified against the SHA1
announced by the minetest server (before loading it and inserting it
into the file cache).
- Bugfix: Only send TOSERVER_RECEIVED_MEDIA when we actually have all
media. This should fix #863.
2013-08-29 05:22:18 +02:00
|
|
|
std::string tpath = m_media[name].path;
|
2012-01-02 12:31:50 +01:00
|
|
|
|
|
|
|
// Read data
|
|
|
|
std::ifstream fis(tpath.c_str(), std::ios_base::binary);
|
2017-08-14 01:06:12 +02:00
|
|
|
if(!fis.good()){
|
2012-03-25 10:50:29 +02:00
|
|
|
errorstream<<"Server::sendRequestedMedia(): Could not open \""
|
2012-01-02 12:31:50 +01:00
|
|
|
<<tpath<<"\" for reading"<<std::endl;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
std::ostringstream tmp_os(std::ios_base::binary);
|
|
|
|
bool bad = false;
|
[Patch 2/4] Network rework: packet writing, sending and cleanups
NetworkPacket.cpp:
* Remove some deprecated functions, we must use streaming interface
* m_data converted from u8* to std::vector<u8>
* Add an exporter to forge packet to Connection object
* implement operator << std::wstring. n
* implement operator << std::string
* dynamic resize when write packet content.
* fix string writing and performances.
* create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Reliability
* Transmit channel
* Implement putRawString for some ugly char (_INIT packet), and use it.
* Many packet read and write migrated
* Implement oldForgePacket to interface writing with current connection
* fix U8/char/bool writing
* fix string writing and performances.
* add some missing functions
* Use v3s16 read instead of reading x,y,z separately
* Add irr::video::SColor support into packets
* Add some missing handlers
* Add a template function to increase offset
* Throw a serialization error on packet reading (must be improved)
PacketFactories:
* Create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Create ClientCommandFactory, used by server to get useful informations about packet processing (sending).
Client.cpp:
* implement NetworkPacket ::Send interface.
* Move packet handlers to a dedicated file
* Remove Client::Send(SharedBuffer)
Server.cpp:
* implement NetworkPacket ::Send interface.
* Rewrite all packets using NetworkPacket
* Move packet handlers to a dedicated file
* Remove Server::Send(SharedBuffer)
ClientIface.cpp:
* Remove sendToAll(SharedBuffer<u8>)
Connection.hpp rework:
* Remove duplicate include
* Remove duplicate negation
* Remove a useless variable
* Improve code performance by using a m_peers_list instead of scanning m_peers map
* Remove Connection::Send(SharedBuffer)
* Fix useafterfree into NetworkPacket Sending
* Remove unused Connection::sendToAll
Test.cpp:
* Remove dead code
* Update tests to use NetworkPackets
Misc:
* add new wrappers to Send packets in client, using NetworkPacket
* Add NetworkPacket methods for Connection
* coding style fix
* dead code since changes cleanup
* Use v3s16 read instead of reading x,y,z separately in some packets
* Use different files to handle packets received by client and server
* Cleanup: Remove useless includes
ok @Zeno-
Tested by @Zeno- @VanessaE and @nerzhul on running servers
2015-01-16 11:37:49 +01:00
|
|
|
for(;;) {
|
2012-01-02 12:31:50 +01:00
|
|
|
char buf[1024];
|
|
|
|
fis.read(buf, 1024);
|
|
|
|
std::streamsize len = fis.gcount();
|
|
|
|
tmp_os.write(buf, len);
|
2012-03-25 10:50:29 +02:00
|
|
|
file_size_bunch_total += len;
|
2012-01-02 12:31:50 +01:00
|
|
|
if(fis.eof())
|
|
|
|
break;
|
[Patch 2/4] Network rework: packet writing, sending and cleanups
NetworkPacket.cpp:
* Remove some deprecated functions, we must use streaming interface
* m_data converted from u8* to std::vector<u8>
* Add an exporter to forge packet to Connection object
* implement operator << std::wstring. n
* implement operator << std::string
* dynamic resize when write packet content.
* fix string writing and performances.
* create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Reliability
* Transmit channel
* Implement putRawString for some ugly char (_INIT packet), and use it.
* Many packet read and write migrated
* Implement oldForgePacket to interface writing with current connection
* fix U8/char/bool writing
* fix string writing and performances.
* add some missing functions
* Use v3s16 read instead of reading x,y,z separately
* Add irr::video::SColor support into packets
* Add some missing handlers
* Add a template function to increase offset
* Throw a serialization error on packet reading (must be improved)
PacketFactories:
* Create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Create ClientCommandFactory, used by server to get useful informations about packet processing (sending).
Client.cpp:
* implement NetworkPacket ::Send interface.
* Move packet handlers to a dedicated file
* Remove Client::Send(SharedBuffer)
Server.cpp:
* implement NetworkPacket ::Send interface.
* Rewrite all packets using NetworkPacket
* Move packet handlers to a dedicated file
* Remove Server::Send(SharedBuffer)
ClientIface.cpp:
* Remove sendToAll(SharedBuffer<u8>)
Connection.hpp rework:
* Remove duplicate include
* Remove duplicate negation
* Remove a useless variable
* Improve code performance by using a m_peers_list instead of scanning m_peers map
* Remove Connection::Send(SharedBuffer)
* Fix useafterfree into NetworkPacket Sending
* Remove unused Connection::sendToAll
Test.cpp:
* Remove dead code
* Update tests to use NetworkPackets
Misc:
* add new wrappers to Send packets in client, using NetworkPacket
* Add NetworkPacket methods for Connection
* coding style fix
* dead code since changes cleanup
* Use v3s16 read instead of reading x,y,z separately in some packets
* Use different files to handle packets received by client and server
* Cleanup: Remove useless includes
ok @Zeno-
Tested by @Zeno- @VanessaE and @nerzhul on running servers
2015-01-16 11:37:49 +01:00
|
|
|
if(!fis.good()) {
|
2012-01-02 12:31:50 +01:00
|
|
|
bad = true;
|
|
|
|
break;
|
2011-11-15 22:58:56 +01:00
|
|
|
}
|
2011-11-15 10:02:47 +01:00
|
|
|
}
|
2017-08-14 01:06:12 +02:00
|
|
|
if (bad) {
|
2012-03-25 10:50:29 +02:00
|
|
|
errorstream<<"Server::sendRequestedMedia(): Failed to read \""
|
Rewrite client media download and support hash-based remote download
Move most of the media-related code in client.cpp into a new class
ClientMediaDownloader (clientmedia.cpp, clientmedia.h). Among other
things, this class does the following things:
- Download [remote_server][sha1] instead of [remote_server][name]. This
is to support servers that provide the same file name with different
contents.
- Initially fetch [remote_server]index.mth. This file should follow the
Minetest Hashset format (currently version 1) and contain a list of SHA1
hashes that exist on the server.
- The list of needed SHA1s is uploaded (via HTTP POST) when index.mth is
requested, so servers can optionally narrow down the list to the needs
of the client.
- If index.mth is missing (HTTP response code 404), we enter compat mode,
fetching [remote_server][name] as before this commit.
- remote_server can now contain multiple servers, separated by commas.
The downloader code attempts to split requests between the different
servers, as permitted by each server's index.mth. If one server claims
to have a file but actually doesn't (or something fails), we ask a
different server that also claims to have it.
- As before, when none of the remote servers provide a particular
file, we download it via the conventional method, i.e. using
the minetest protocol: TOSERVER_REQUEST_MEDIA / TOCLIENT_MEDIA.
- Bugfix: Every downloaded file's SHA1 is now verified against the SHA1
announced by the minetest server (before loading it and inserting it
into the file cache).
- Bugfix: Only send TOSERVER_RECEIVED_MEDIA when we actually have all
media. This should fix #863.
2013-08-29 05:22:18 +02:00
|
|
|
<<name<<"\""<<std::endl;
|
2012-01-02 12:31:50 +01:00
|
|
|
continue;
|
|
|
|
}
|
2012-03-25 10:50:29 +02:00
|
|
|
/*infostream<<"Server::sendRequestedMedia(): Loaded \""
|
2012-01-02 12:31:50 +01:00
|
|
|
<<tname<<"\""<<std::endl;*/
|
|
|
|
// Put in list
|
2017-08-14 01:06:12 +02:00
|
|
|
file_bunches[file_bunches.size()-1].emplace_back(name, tpath, tmp_os.str());
|
2012-01-02 12:31:50 +01:00
|
|
|
|
|
|
|
// Start next bunch if got enough data
|
[Patch 2/4] Network rework: packet writing, sending and cleanups
NetworkPacket.cpp:
* Remove some deprecated functions, we must use streaming interface
* m_data converted from u8* to std::vector<u8>
* Add an exporter to forge packet to Connection object
* implement operator << std::wstring. n
* implement operator << std::string
* dynamic resize when write packet content.
* fix string writing and performances.
* create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Reliability
* Transmit channel
* Implement putRawString for some ugly char (_INIT packet), and use it.
* Many packet read and write migrated
* Implement oldForgePacket to interface writing with current connection
* fix U8/char/bool writing
* fix string writing and performances.
* add some missing functions
* Use v3s16 read instead of reading x,y,z separately
* Add irr::video::SColor support into packets
* Add some missing handlers
* Add a template function to increase offset
* Throw a serialization error on packet reading (must be improved)
PacketFactories:
* Create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Create ClientCommandFactory, used by server to get useful informations about packet processing (sending).
Client.cpp:
* implement NetworkPacket ::Send interface.
* Move packet handlers to a dedicated file
* Remove Client::Send(SharedBuffer)
Server.cpp:
* implement NetworkPacket ::Send interface.
* Rewrite all packets using NetworkPacket
* Move packet handlers to a dedicated file
* Remove Server::Send(SharedBuffer)
ClientIface.cpp:
* Remove sendToAll(SharedBuffer<u8>)
Connection.hpp rework:
* Remove duplicate include
* Remove duplicate negation
* Remove a useless variable
* Improve code performance by using a m_peers_list instead of scanning m_peers map
* Remove Connection::Send(SharedBuffer)
* Fix useafterfree into NetworkPacket Sending
* Remove unused Connection::sendToAll
Test.cpp:
* Remove dead code
* Update tests to use NetworkPackets
Misc:
* add new wrappers to Send packets in client, using NetworkPacket
* Add NetworkPacket methods for Connection
* coding style fix
* dead code since changes cleanup
* Use v3s16 read instead of reading x,y,z separately in some packets
* Use different files to handle packets received by client and server
* Cleanup: Remove useless includes
ok @Zeno-
Tested by @Zeno- @VanessaE and @nerzhul on running servers
2015-01-16 11:37:49 +01:00
|
|
|
if(file_size_bunch_total >= bytes_per_bunch) {
|
2017-08-14 01:06:12 +02:00
|
|
|
file_bunches.emplace_back();
|
2012-03-25 10:50:29 +02:00
|
|
|
file_size_bunch_total = 0;
|
2012-01-02 12:31:50 +01:00
|
|
|
}
|
|
|
|
|
2011-11-15 10:02:47 +01:00
|
|
|
}
|
|
|
|
|
2011-11-15 22:58:56 +01:00
|
|
|
/* Create and send packets */
|
2012-01-02 12:31:50 +01:00
|
|
|
|
[Patch 2/4] Network rework: packet writing, sending and cleanups
NetworkPacket.cpp:
* Remove some deprecated functions, we must use streaming interface
* m_data converted from u8* to std::vector<u8>
* Add an exporter to forge packet to Connection object
* implement operator << std::wstring. n
* implement operator << std::string
* dynamic resize when write packet content.
* fix string writing and performances.
* create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Reliability
* Transmit channel
* Implement putRawString for some ugly char (_INIT packet), and use it.
* Many packet read and write migrated
* Implement oldForgePacket to interface writing with current connection
* fix U8/char/bool writing
* fix string writing and performances.
* add some missing functions
* Use v3s16 read instead of reading x,y,z separately
* Add irr::video::SColor support into packets
* Add some missing handlers
* Add a template function to increase offset
* Throw a serialization error on packet reading (must be improved)
PacketFactories:
* Create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Create ClientCommandFactory, used by server to get useful informations about packet processing (sending).
Client.cpp:
* implement NetworkPacket ::Send interface.
* Move packet handlers to a dedicated file
* Remove Client::Send(SharedBuffer)
Server.cpp:
* implement NetworkPacket ::Send interface.
* Rewrite all packets using NetworkPacket
* Move packet handlers to a dedicated file
* Remove Server::Send(SharedBuffer)
ClientIface.cpp:
* Remove sendToAll(SharedBuffer<u8>)
Connection.hpp rework:
* Remove duplicate include
* Remove duplicate negation
* Remove a useless variable
* Improve code performance by using a m_peers_list instead of scanning m_peers map
* Remove Connection::Send(SharedBuffer)
* Fix useafterfree into NetworkPacket Sending
* Remove unused Connection::sendToAll
Test.cpp:
* Remove dead code
* Update tests to use NetworkPackets
Misc:
* add new wrappers to Send packets in client, using NetworkPacket
* Add NetworkPacket methods for Connection
* coding style fix
* dead code since changes cleanup
* Use v3s16 read instead of reading x,y,z separately in some packets
* Use different files to handle packets received by client and server
* Cleanup: Remove useless includes
ok @Zeno-
Tested by @Zeno- @VanessaE and @nerzhul on running servers
2015-01-16 11:37:49 +01:00
|
|
|
u16 num_bunches = file_bunches.size();
|
2017-08-14 01:06:12 +02:00
|
|
|
for (u16 i = 0; i < num_bunches; i++) {
|
2012-03-25 10:50:29 +02:00
|
|
|
/*
|
|
|
|
u16 command
|
|
|
|
u16 total number of texture bunches
|
|
|
|
u16 index of this bunch
|
|
|
|
u32 number of files in this bunch
|
|
|
|
for each file {
|
|
|
|
u16 length of name
|
|
|
|
string name
|
|
|
|
u32 length of data
|
|
|
|
data
|
2011-11-15 22:58:56 +01:00
|
|
|
}
|
2012-03-25 10:50:29 +02:00
|
|
|
*/
|
2011-11-15 10:02:47 +01:00
|
|
|
|
2015-03-13 22:01:49 +01:00
|
|
|
NetworkPacket pkt(TOCLIENT_MEDIA, 4 + 0, peer_id);
|
|
|
|
pkt << num_bunches << i << (u32) file_bunches[i].size();
|
2012-01-02 12:31:50 +01:00
|
|
|
|
2017-08-14 01:06:12 +02:00
|
|
|
for (const SendableMedia &j : file_bunches[i]) {
|
|
|
|
pkt << j.name;
|
|
|
|
pkt.putLongString(j.data);
|
2012-03-25 10:50:29 +02:00
|
|
|
}
|
2012-01-02 12:31:50 +01:00
|
|
|
|
[Patch 2/4] Network rework: packet writing, sending and cleanups
NetworkPacket.cpp:
* Remove some deprecated functions, we must use streaming interface
* m_data converted from u8* to std::vector<u8>
* Add an exporter to forge packet to Connection object
* implement operator << std::wstring. n
* implement operator << std::string
* dynamic resize when write packet content.
* fix string writing and performances.
* create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Reliability
* Transmit channel
* Implement putRawString for some ugly char (_INIT packet), and use it.
* Many packet read and write migrated
* Implement oldForgePacket to interface writing with current connection
* fix U8/char/bool writing
* fix string writing and performances.
* add some missing functions
* Use v3s16 read instead of reading x,y,z separately
* Add irr::video::SColor support into packets
* Add some missing handlers
* Add a template function to increase offset
* Throw a serialization error on packet reading (must be improved)
PacketFactories:
* Create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Create ClientCommandFactory, used by server to get useful informations about packet processing (sending).
Client.cpp:
* implement NetworkPacket ::Send interface.
* Move packet handlers to a dedicated file
* Remove Client::Send(SharedBuffer)
Server.cpp:
* implement NetworkPacket ::Send interface.
* Rewrite all packets using NetworkPacket
* Move packet handlers to a dedicated file
* Remove Server::Send(SharedBuffer)
ClientIface.cpp:
* Remove sendToAll(SharedBuffer<u8>)
Connection.hpp rework:
* Remove duplicate include
* Remove duplicate negation
* Remove a useless variable
* Improve code performance by using a m_peers_list instead of scanning m_peers map
* Remove Connection::Send(SharedBuffer)
* Fix useafterfree into NetworkPacket Sending
* Remove unused Connection::sendToAll
Test.cpp:
* Remove dead code
* Update tests to use NetworkPackets
Misc:
* add new wrappers to Send packets in client, using NetworkPacket
* Add NetworkPacket methods for Connection
* coding style fix
* dead code since changes cleanup
* Use v3s16 read instead of reading x,y,z separately in some packets
* Use different files to handle packets received by client and server
* Cleanup: Remove useless includes
ok @Zeno-
Tested by @Zeno- @VanessaE and @nerzhul on running servers
2015-01-16 11:37:49 +01:00
|
|
|
verbosestream << "Server::sendRequestedMedia(): bunch "
|
|
|
|
<< i << "/" << num_bunches
|
|
|
|
<< " files=" << file_bunches[i].size()
|
2015-03-13 22:01:49 +01:00
|
|
|
<< " size=" << pkt.getSize() << std::endl;
|
|
|
|
Send(&pkt);
|
2012-03-25 10:50:29 +02:00
|
|
|
}
|
2011-11-15 10:02:47 +01:00
|
|
|
}
|
|
|
|
|
2017-09-27 19:47:36 +02:00
|
|
|
void Server::sendDetachedInventory(const std::string &name, session_t peer_id)
|
2012-07-24 19:57:17 +02:00
|
|
|
{
|
2018-10-10 20:48:58 +02:00
|
|
|
const auto &inv_it = m_detached_inventories.find(name);
|
|
|
|
const auto &player_it = m_detached_inventories_player.find(name);
|
[Patch 2/4] Network rework: packet writing, sending and cleanups
NetworkPacket.cpp:
* Remove some deprecated functions, we must use streaming interface
* m_data converted from u8* to std::vector<u8>
* Add an exporter to forge packet to Connection object
* implement operator << std::wstring. n
* implement operator << std::string
* dynamic resize when write packet content.
* fix string writing and performances.
* create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Reliability
* Transmit channel
* Implement putRawString for some ugly char (_INIT packet), and use it.
* Many packet read and write migrated
* Implement oldForgePacket to interface writing with current connection
* fix U8/char/bool writing
* fix string writing and performances.
* add some missing functions
* Use v3s16 read instead of reading x,y,z separately
* Add irr::video::SColor support into packets
* Add some missing handlers
* Add a template function to increase offset
* Throw a serialization error on packet reading (must be improved)
PacketFactories:
* Create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Create ClientCommandFactory, used by server to get useful informations about packet processing (sending).
Client.cpp:
* implement NetworkPacket ::Send interface.
* Move packet handlers to a dedicated file
* Remove Client::Send(SharedBuffer)
Server.cpp:
* implement NetworkPacket ::Send interface.
* Rewrite all packets using NetworkPacket
* Move packet handlers to a dedicated file
* Remove Server::Send(SharedBuffer)
ClientIface.cpp:
* Remove sendToAll(SharedBuffer<u8>)
Connection.hpp rework:
* Remove duplicate include
* Remove duplicate negation
* Remove a useless variable
* Improve code performance by using a m_peers_list instead of scanning m_peers map
* Remove Connection::Send(SharedBuffer)
* Fix useafterfree into NetworkPacket Sending
* Remove unused Connection::sendToAll
Test.cpp:
* Remove dead code
* Update tests to use NetworkPackets
Misc:
* add new wrappers to Send packets in client, using NetworkPacket
* Add NetworkPacket methods for Connection
* coding style fix
* dead code since changes cleanup
* Use v3s16 read instead of reading x,y,z separately in some packets
* Use different files to handle packets received by client and server
* Cleanup: Remove useless includes
ok @Zeno-
Tested by @Zeno- @VanessaE and @nerzhul on running servers
2015-01-16 11:37:49 +01:00
|
|
|
|
2018-10-10 20:48:58 +02:00
|
|
|
if (player_it == m_detached_inventories_player.end() ||
|
|
|
|
player_it->second.empty()) {
|
|
|
|
// OK. Send to everyone
|
|
|
|
} else {
|
|
|
|
RemotePlayer *p = m_env->getPlayer(player_it->second.c_str());
|
|
|
|
if (!p)
|
|
|
|
return; // Player is offline
|
2012-07-24 19:57:17 +02:00
|
|
|
|
2018-10-10 20:48:58 +02:00
|
|
|
if (peer_id != PEER_ID_INEXISTENT && peer_id != p->getPeerId())
|
|
|
|
return; // Caller requested send to a different player, so don't send.
|
|
|
|
|
|
|
|
peer_id = p->getPeerId();
|
|
|
|
}
|
2012-07-24 19:57:17 +02:00
|
|
|
|
2015-03-13 22:01:49 +01:00
|
|
|
NetworkPacket pkt(TOCLIENT_DETACHED_INVENTORY, 0, peer_id);
|
2018-10-10 20:48:58 +02:00
|
|
|
pkt << name;
|
[Patch 2/4] Network rework: packet writing, sending and cleanups
NetworkPacket.cpp:
* Remove some deprecated functions, we must use streaming interface
* m_data converted from u8* to std::vector<u8>
* Add an exporter to forge packet to Connection object
* implement operator << std::wstring. n
* implement operator << std::string
* dynamic resize when write packet content.
* fix string writing and performances.
* create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Reliability
* Transmit channel
* Implement putRawString for some ugly char (_INIT packet), and use it.
* Many packet read and write migrated
* Implement oldForgePacket to interface writing with current connection
* fix U8/char/bool writing
* fix string writing and performances.
* add some missing functions
* Use v3s16 read instead of reading x,y,z separately
* Add irr::video::SColor support into packets
* Add some missing handlers
* Add a template function to increase offset
* Throw a serialization error on packet reading (must be improved)
PacketFactories:
* Create ServerCommandFactory, used by client to get useful informations about packet processing (sending).
* Create ClientCommandFactory, used by server to get useful informations about packet processing (sending).
Client.cpp:
* implement NetworkPacket ::Send interface.
* Move packet handlers to a dedicated file
* Remove Client::Send(SharedBuffer)
Server.cpp:
* implement NetworkPacket ::Send interface.
* Rewrite all packets using NetworkPacket
* Move packet handlers to a dedicated file
* Remove Server::Send(SharedBuffer)
ClientIface.cpp:
* Remove sendToAll(SharedBuffer<u8>)
Connection.hpp rework:
* Remove duplicate include
* Remove duplicate negation
* Remove a useless variable
* Improve code performance by using a m_peers_list instead of scanning m_peers map
* Remove Connection::Send(SharedBuffer)
* Fix useafterfree into NetworkPacket Sending
* Remove unused Connection::sendToAll
Test.cpp:
* Remove dead code
* Update tests to use NetworkPackets
Misc:
* add new wrappers to Send packets in client, using NetworkPacket
* Add NetworkPacket methods for Connection
* coding style fix
* dead code since changes cleanup
* Use v3s16 read instead of reading x,y,z separately in some packets
* Use different files to handle packets received by client and server
* Cleanup: Remove useless includes
ok @Zeno-
Tested by @Zeno- @VanessaE and @nerzhul on running servers
2015-01-16 11:37:49 +01:00
|
|
|
|
2018-10-10 20:48:58 +02:00
|
|
|
if (inv_it == m_detached_inventories.end()) {
|
|
|
|
pkt << false; // Remove inventory
|
2016-11-26 17:35:25 +01:00
|
|
|
} else {
|
2018-10-10 20:48:58 +02:00
|
|
|
pkt << true; // Update inventory
|
|
|
|
|
|
|
|
// Serialization & NetworkPacket isn't a love story
|
|
|
|
std::ostringstream os(std::ios_base::binary);
|
|
|
|
inv_it->second->serialize(os);
|
2019-03-07 07:41:21 +01:00
|
|
|
|
|
|
|
std::string os_str = os.str();
|
|
|
|
pkt << static_cast<u16>(os_str.size()); // HACK: to keep compatibility with 5.0.0 clients
|
|
|
|
pkt.putRawString(os_str);
|
2012-07-24 19:57:17 +02:00
|
|
|
}
|
2018-10-10 20:48:58 +02:00
|
|
|
|
|
|
|
if (peer_id == PEER_ID_INEXISTENT)
|
|
|
|
m_clients.sendToAll(&pkt);
|
|
|
|
else
|
|
|
|
Send(&pkt);
|
2012-07-24 19:57:17 +02:00
|
|
|
}
|
|
|
|
|
2017-09-27 19:47:36 +02:00
|
|
|
void Server::sendDetachedInventories(session_t peer_id)
|
2012-07-24 19:57:17 +02:00
|
|
|
{
|
2017-08-14 01:06:12 +02:00
|
|
|
for (const auto &detached_inventory : m_detached_inventories) {
|
|
|
|
const std::string &name = detached_inventory.first;
|
2012-07-24 19:57:17 +02:00
|
|
|
//Inventory *inv = i->second;
|
|
|
|
sendDetachedInventory(name, peer_id);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-04-21 18:35:17 +02:00
|
|
|
/*
|
|
|
|
Something random
|
|
|
|
*/
|
|
|
|
|
2018-03-28 17:05:18 +02:00
|
|
|
void Server::DiePlayer(session_t peer_id, const PlayerHPChangeReason &reason)
|
2011-10-15 13:46:59 +02:00
|
|
|
{
|
2012-03-19 03:04:16 +01:00
|
|
|
PlayerSAO *playersao = getPlayerSAO(peer_id);
|
2016-05-08 11:02:11 +02:00
|
|
|
// In some rare cases this can be NULL -- if the player is disconnected
|
|
|
|
// when a Lua function modifies l_punch, for example
|
|
|
|
if (!playersao)
|
|
|
|
return;
|
2011-12-01 22:33:48 +01:00
|
|
|
|
2015-03-02 17:31:31 +01:00
|
|
|
infostream << "Server::DiePlayer(): Player "
|
|
|
|
<< playersao->getPlayer()->getName()
|
|
|
|
<< " dies" << std::endl;
|
2012-03-19 03:04:16 +01:00
|
|
|
|
2018-03-28 17:05:18 +02:00
|
|
|
playersao->setHP(0, reason);
|
2018-04-30 18:43:49 +02:00
|
|
|
playersao->clearParentAttachment();
|
2012-03-19 03:04:16 +01:00
|
|
|
|
2012-01-05 23:12:33 +01:00
|
|
|
// Trigger scripted stuff
|
2018-03-28 17:05:18 +02:00
|
|
|
m_script->on_dieplayer(playersao, reason);
|
2011-12-02 16:19:42 +01:00
|
|
|
|
2012-03-19 03:04:16 +01:00
|
|
|
SendPlayerHP(peer_id);
|
2014-01-31 00:24:00 +01:00
|
|
|
SendDeathscreen(peer_id, false, v3f(0,0,0));
|
2011-10-15 13:46:59 +02:00
|
|
|
}
|
|
|
|
|
2017-09-27 19:47:36 +02:00
|
|
|
void Server::RespawnPlayer(session_t peer_id)
|
2011-10-15 13:46:59 +02:00
|
|
|
{
|
2012-03-19 03:04:16 +01:00
|
|
|
PlayerSAO *playersao = getPlayerSAO(peer_id);
|
|
|
|
assert(playersao);
|
|
|
|
|
2015-03-02 17:31:31 +01:00
|
|
|
infostream << "Server::RespawnPlayer(): Player "
|
|
|
|
<< playersao->getPlayer()->getName()
|
|
|
|
<< " respawns" << std::endl;
|
2012-03-19 03:04:16 +01:00
|
|
|
|
2018-03-28 17:05:18 +02:00
|
|
|
playersao->setHP(playersao->accessObjectProperties()->hp_max,
|
|
|
|
PlayerHPChangeReason(PlayerHPChangeReason::RESPAWN));
|
2017-09-15 12:18:47 +02:00
|
|
|
playersao->setBreath(playersao->accessObjectProperties()->breath_max);
|
2012-03-19 03:04:16 +01:00
|
|
|
|
2013-05-25 00:51:02 +02:00
|
|
|
bool repositioned = m_script->on_respawnplayer(playersao);
|
2016-12-22 19:29:15 +01:00
|
|
|
if (!repositioned) {
|
2015-03-04 12:19:26 +01:00
|
|
|
// setPos will send the new position to client
|
2017-04-23 14:35:08 +02:00
|
|
|
playersao->setPos(findSpawnPos());
|
2011-11-26 02:20:19 +01:00
|
|
|
}
|
2016-12-22 19:29:15 +01:00
|
|
|
|
|
|
|
SendPlayerHP(peer_id);
|
2011-10-15 13:46:59 +02:00
|
|
|
}
|
2015-07-17 16:40:41 +02:00
|
|
|
|
|
|
|
|
2017-09-27 19:47:36 +02:00
|
|
|
void Server::DenySudoAccess(session_t peer_id)
|
2015-04-12 04:49:13 +02:00
|
|
|
{
|
|
|
|
NetworkPacket pkt(TOCLIENT_DENY_SUDO_MODE, 0, peer_id);
|
|
|
|
Send(&pkt);
|
|
|
|
}
|
2011-10-15 13:46:59 +02:00
|
|
|
|
2015-07-17 16:40:41 +02:00
|
|
|
|
2017-09-27 19:47:36 +02:00
|
|
|
void Server::DenyAccessVerCompliant(session_t peer_id, u16 proto_ver, AccessDeniedCode reason,
|
2015-07-17 16:40:41 +02:00
|
|
|
const std::string &str_reason, bool reconnect)
|
|
|
|
{
|
2017-09-12 19:48:09 +02:00
|
|
|
SendAccessDenied(peer_id, reason, str_reason, reconnect);
|
2015-07-17 16:40:41 +02:00
|
|
|
|
|
|
|
m_clients.event(peer_id, CSE_SetDenied);
|
2017-09-26 00:11:20 +02:00
|
|
|
DisconnectPeer(peer_id);
|
2015-07-17 16:40:41 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-09-27 19:47:36 +02:00
|
|
|
void Server::DenyAccess(session_t peer_id, AccessDeniedCode reason,
|
|
|
|
const std::string &custom_reason)
|
2015-03-13 16:35:21 +01:00
|
|
|
{
|
|
|
|
SendAccessDenied(peer_id, reason, custom_reason);
|
|
|
|
m_clients.event(peer_id, CSE_SetDenied);
|
2017-09-26 00:11:20 +02:00
|
|
|
DisconnectPeer(peer_id);
|
2015-03-13 16:35:21 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// 13/03/15: remove this function when protocol version 25 will become
|
|
|
|
// the minimum version for MT users, maybe in 1 year
|
2017-09-27 19:47:36 +02:00
|
|
|
void Server::DenyAccess_Legacy(session_t peer_id, const std::wstring &reason)
|
2013-08-04 07:17:07 +02:00
|
|
|
{
|
2015-03-13 16:35:21 +01:00
|
|
|
SendAccessDenied_Legacy(peer_id, reason);
|
2014-06-28 08:02:38 +02:00
|
|
|
m_clients.event(peer_id, CSE_SetDenied);
|
2017-09-26 00:11:20 +02:00
|
|
|
DisconnectPeer(peer_id);
|
|
|
|
}
|
|
|
|
|
2017-09-27 19:47:36 +02:00
|
|
|
void Server::DisconnectPeer(session_t peer_id)
|
2017-09-26 00:11:20 +02:00
|
|
|
{
|
|
|
|
m_modchannel_mgr->leaveAllChannels(peer_id);
|
2017-08-24 08:28:54 +02:00
|
|
|
m_con->DisconnectPeer(peer_id);
|
2013-08-04 07:17:07 +02:00
|
|
|
}
|
|
|
|
|
2017-09-27 19:47:36 +02:00
|
|
|
void Server::acceptAuth(session_t peer_id, bool forSudoMode)
|
2015-04-12 04:49:13 +02:00
|
|
|
{
|
|
|
|
if (!forSudoMode) {
|
|
|
|
RemoteClient* client = getClient(peer_id, CS_Invalid);
|
|
|
|
|
|
|
|
NetworkPacket resp_pkt(TOCLIENT_AUTH_ACCEPT, 1 + 6 + 8 + 4, peer_id);
|
|
|
|
|
|
|
|
// Right now, the auth mechs don't change between login and sudo mode.
|
|
|
|
u32 sudo_auth_mechs = client->allowed_auth_mechs;
|
|
|
|
client->allowed_sudo_mechs = sudo_auth_mechs;
|
|
|
|
|
|
|
|
resp_pkt << v3f(0,0,0) << (u64) m_env->getServerMap().getSeed()
|
|
|
|
<< g_settings->getFloat("dedicated_server_step")
|
|
|
|
<< sudo_auth_mechs;
|
|
|
|
|
|
|
|
Send(&resp_pkt);
|
|
|
|
m_clients.event(peer_id, CSE_AuthAccept);
|
|
|
|
} else {
|
|
|
|
NetworkPacket resp_pkt(TOCLIENT_ACCEPT_SUDO_MODE, 1 + 6 + 8 + 4, peer_id);
|
|
|
|
|
|
|
|
// We only support SRP right now
|
|
|
|
u32 sudo_auth_mechs = AUTH_MECHANISM_FIRST_SRP;
|
|
|
|
|
|
|
|
resp_pkt << sudo_auth_mechs;
|
|
|
|
Send(&resp_pkt);
|
|
|
|
m_clients.event(peer_id, CSE_SudoSuccess);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-09-27 19:47:36 +02:00
|
|
|
void Server::DeleteClient(session_t peer_id, ClientDeletionReason reason)
|
2013-08-04 07:17:07 +02:00
|
|
|
{
|
2014-01-31 00:24:00 +01:00
|
|
|
std::wstring message;
|
2013-08-04 07:17:07 +02:00
|
|
|
{
|
2014-01-31 00:24:00 +01:00
|
|
|
/*
|
|
|
|
Clear references to playing sounds
|
|
|
|
*/
|
2017-06-04 21:00:04 +02:00
|
|
|
for (std::unordered_map<s32, ServerPlayingSound>::iterator
|
2016-10-06 19:20:12 +02:00
|
|
|
i = m_playing_sounds.begin(); i != m_playing_sounds.end();) {
|
2014-01-31 00:24:00 +01:00
|
|
|
ServerPlayingSound &psound = i->second;
|
|
|
|
psound.clients.erase(peer_id);
|
2016-10-06 19:20:12 +02:00
|
|
|
if (psound.clients.empty())
|
2014-01-31 00:24:00 +01:00
|
|
|
m_playing_sounds.erase(i++);
|
|
|
|
else
|
2015-08-25 22:23:05 +02:00
|
|
|
++i;
|
2014-01-31 00:24:00 +01:00
|
|
|
}
|
2013-08-04 07:17:07 +02:00
|
|
|
|
2018-02-18 22:33:42 +01:00
|
|
|
// clear formspec info so the next client can't abuse the current state
|
|
|
|
m_formspec_state_data.erase(peer_id);
|
|
|
|
|
2016-10-08 12:21:41 +02:00
|
|
|
RemotePlayer *player = m_env->getPlayer(peer_id);
|
2013-08-04 07:17:07 +02:00
|
|
|
|
2014-01-31 00:24:00 +01:00
|
|
|
/* Run scripts and remove from environment */
|
2017-08-15 09:30:31 +02:00
|
|
|
if (player) {
|
2016-10-08 10:38:04 +02:00
|
|
|
PlayerSAO *playersao = player->getPlayerSAO();
|
|
|
|
assert(playersao);
|
2013-08-04 07:17:07 +02:00
|
|
|
|
2018-05-12 10:55:43 +02:00
|
|
|
playersao->clearChildAttachments();
|
|
|
|
playersao->clearParentAttachment();
|
|
|
|
|
2017-06-08 15:30:09 +02:00
|
|
|
// inform connected clients
|
2018-10-10 20:48:58 +02:00
|
|
|
const std::string &player_name = player->getName();
|
2017-06-08 15:30:09 +02:00
|
|
|
NetworkPacket notice(TOCLIENT_UPDATE_PLAYER_LIST, 0, PEER_ID_INEXISTENT);
|
|
|
|
// (u16) 1 + std::string represents a vector serialization representation
|
2018-10-10 20:48:58 +02:00
|
|
|
notice << (u8) PLAYER_LIST_REMOVE << (u16) 1 << player_name;
|
2017-06-08 15:30:09 +02:00
|
|
|
m_clients.sendToAll(¬ice);
|
|
|
|
// run scripts
|
2016-10-08 10:38:04 +02:00
|
|
|
m_script->on_leaveplayer(playersao, reason == CDR_TIMEOUT);
|
2013-08-04 07:17:07 +02:00
|
|
|
|
2016-10-08 10:38:04 +02:00
|
|
|
playersao->disconnected();
|
2013-08-04 07:17:07 +02:00
|
|
|
}
|
|
|
|
|
2014-01-31 00:24:00 +01:00
|
|
|
/*
|
|
|
|
Print out action
|
|
|
|
*/
|
2013-08-04 07:17:07 +02:00
|
|
|
{
|
2017-08-15 09:30:31 +02:00
|
|
|
if (player && reason != CDR_DENY) {
|
2014-01-31 00:24:00 +01:00
|
|
|
std::ostringstream os(std::ios_base::binary);
|
2017-09-27 23:48:06 +02:00
|
|
|
std::vector<session_t> clients = m_clients.getClientIDs();
|
2014-01-31 00:24:00 +01:00
|
|
|
|
2017-09-27 23:48:06 +02:00
|
|
|
for (const session_t client_id : clients) {
|
2014-01-31 00:24:00 +01:00
|
|
|
// Get player
|
2017-08-14 01:06:12 +02:00
|
|
|
RemotePlayer *player = m_env->getPlayer(client_id);
|
2016-10-08 16:31:22 +02:00
|
|
|
if (!player)
|
2014-01-31 00:24:00 +01:00
|
|
|
continue;
|
2015-03-04 16:30:24 +01:00
|
|
|
|
2014-01-31 00:24:00 +01:00
|
|
|
// Get name of player
|
2015-03-04 16:30:24 +01:00
|
|
|
os << player->getName() << " ";
|
2014-01-31 00:24:00 +01:00
|
|
|
}
|
2013-08-04 07:17:07 +02:00
|
|
|
|
2015-07-25 07:43:32 +02:00
|
|
|
std::string name = player->getName();
|
|
|
|
actionstream << name << " "
|
2015-03-04 16:30:24 +01:00
|
|
|
<< (reason == CDR_TIMEOUT ? "times out." : "leaves game.")
|
|
|
|
<< " List of players: " << os.str() << std::endl;
|
2015-07-25 07:43:32 +02:00
|
|
|
if (m_admin_chat)
|
|
|
|
m_admin_chat->outgoing_queue.push_back(
|
|
|
|
new ChatEventNick(CET_NICK_REMOVE, name));
|
2014-01-31 00:24:00 +01:00
|
|
|
}
|
2013-08-04 07:17:07 +02:00
|
|
|
}
|
2014-02-12 19:47:27 +01:00
|
|
|
{
|
2015-04-07 12:13:12 +02:00
|
|
|
MutexAutoLock env_lock(m_env_mutex);
|
2014-02-12 19:47:27 +01:00
|
|
|
m_clients.DeleteClient(peer_id);
|
|
|
|
}
|
2013-08-04 07:17:07 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Send leave chat message to all remaining clients
|
2017-07-16 10:47:31 +02:00
|
|
|
if (!message.empty()) {
|
|
|
|
SendChatMessage(PEER_ID_INEXISTENT,
|
|
|
|
ChatMessage(CHATMESSAGE_TYPE_ANNOUNCE, message));
|
|
|
|
}
|
2013-08-04 07:17:07 +02:00
|
|
|
}
|
|
|
|
|
2016-10-08 19:08:23 +02:00
|
|
|
void Server::UpdateCrafting(RemotePlayer *player)
|
2011-04-21 18:35:17 +02:00
|
|
|
{
|
2018-05-05 22:37:07 +02:00
|
|
|
InventoryList *clist = player->inventory.getList("craft");
|
|
|
|
if (!clist || clist->getSize() == 0)
|
|
|
|
return;
|
|
|
|
|
2012-01-21 21:21:41 +01:00
|
|
|
// Get a preview for crafting
|
|
|
|
ItemStack preview;
|
2013-10-26 11:25:28 +02:00
|
|
|
InventoryLocation loc;
|
|
|
|
loc.setPlayer(player->getName());
|
2015-06-02 20:30:04 +02:00
|
|
|
std::vector<ItemStack> output_replacements;
|
|
|
|
getCraftingResult(&player->inventory, preview, output_replacements, false, this);
|
2016-10-08 10:38:04 +02:00
|
|
|
m_env->getScriptIface()->item_CraftPredict(preview, player->getPlayerSAO(),
|
2018-05-05 22:37:07 +02:00
|
|
|
clist, loc);
|
2012-01-21 21:21:41 +01:00
|
|
|
|
|
|
|
InventoryList *plist = player->inventory.getList("craftpreview");
|
2018-05-05 22:37:07 +02:00
|
|
|
if (plist && plist->getSize() >= 1) {
|
|
|
|
// Put the new preview in
|
|
|
|
plist->changeItem(0, preview);
|
|
|
|
}
|
2010-11-27 00:02:21 +01:00
|
|
|
}
|
|
|
|
|
2015-11-09 06:04:24 +01:00
|
|
|
void Server::handleChatInterfaceEvent(ChatEvent *evt)
|
|
|
|
{
|
|
|
|
if (evt->type == CET_NICK_ADD) {
|
|
|
|
// The terminal informed us of its nick choice
|
|
|
|
m_admin_nick = ((ChatEventNick *)evt)->nick;
|
|
|
|
if (!m_script->getAuth(m_admin_nick, NULL, NULL)) {
|
|
|
|
errorstream << "You haven't set up an account." << std::endl
|
|
|
|
<< "Please log in using the client as '"
|
|
|
|
<< m_admin_nick << "' with a secure password." << std::endl
|
|
|
|
<< "Until then, you can't execute admin tasks via the console," << std::endl
|
|
|
|
<< "and everybody can claim the user account instead of you," << std::endl
|
|
|
|
<< "giving them full control over this server." << std::endl;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
assert(evt->type == CET_CHAT);
|
|
|
|
handleAdminChat((ChatEventChat *)evt);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-07-25 07:43:32 +02:00
|
|
|
std::wstring Server::handleChat(const std::string &name, const std::wstring &wname,
|
2017-06-10 13:49:44 +02:00
|
|
|
std::wstring wmessage, bool check_shout_priv, RemotePlayer *player)
|
2015-07-25 07:43:32 +02:00
|
|
|
{
|
|
|
|
// If something goes wrong, this player is to blame
|
|
|
|
RollbackScopeActor rollback_scope(m_rollback,
|
|
|
|
std::string("player:") + name);
|
|
|
|
|
2017-06-10 13:49:44 +02:00
|
|
|
if (g_settings->getBool("strip_color_codes"))
|
|
|
|
wmessage = unescape_enriched(wmessage);
|
|
|
|
|
2016-10-06 03:32:52 +02:00
|
|
|
if (player) {
|
|
|
|
switch (player->canSendChatMessage()) {
|
|
|
|
case RPLAYER_CHATRESULT_FLOODING: {
|
|
|
|
std::wstringstream ws;
|
|
|
|
ws << L"You cannot send more messages. You are limited to "
|
2017-01-25 10:57:33 +01:00
|
|
|
<< g_settings->getFloat("chat_message_limit_per_10sec")
|
|
|
|
<< L" messages per 10 seconds.";
|
2016-10-06 03:32:52 +02:00
|
|
|
return ws.str();
|
|
|
|
}
|
|
|
|
case RPLAYER_CHATRESULT_KICK:
|
2017-09-30 12:00:05 +02:00
|
|
|
DenyAccess_Legacy(player->getPeerId(),
|
2017-01-25 10:57:33 +01:00
|
|
|
L"You have been kicked due to message flooding.");
|
2016-10-06 03:32:52 +02:00
|
|
|
return L"";
|
2017-01-25 10:57:33 +01:00
|
|
|
case RPLAYER_CHATRESULT_OK:
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
FATAL_ERROR("Unhandled chat filtering result found.");
|
2016-10-04 18:17:12 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-01-25 10:57:33 +01:00
|
|
|
if (m_max_chatmessage_length > 0
|
|
|
|
&& wmessage.length() > m_max_chatmessage_length) {
|
2016-10-04 18:17:12 +02:00
|
|
|
return L"Your message exceed the maximum chat message limit set on the server. "
|
2017-01-25 10:57:33 +01:00
|
|
|
L"It was refused. Send a shorter message";
|
2016-10-04 18:17:12 +02:00
|
|
|
}
|
|
|
|
|
2017-01-25 10:57:33 +01:00
|
|
|
// Run script hook, exit if script ate the chat message
|
|
|
|
if (m_script->on_chat_message(name, wide_to_utf8(wmessage)))
|
|
|
|
return L"";
|
|
|
|
|
|
|
|
// Line to send
|
|
|
|
std::wstring line;
|
|
|
|
// Whether to send line to the player that sent the message, or to all players
|
|
|
|
bool broadcast_line = true;
|
|
|
|
|
2017-04-08 20:03:57 +02:00
|
|
|
if (check_shout_priv && !checkPriv(name, "shout")) {
|
|
|
|
line += L"-!- You don't have permission to shout.";
|
2015-07-25 07:43:32 +02:00
|
|
|
broadcast_line = false;
|
|
|
|
} else {
|
2017-04-08 20:03:57 +02:00
|
|
|
line += L"<";
|
|
|
|
line += wname;
|
|
|
|
line += L"> ";
|
|
|
|
line += wmessage;
|
2015-07-25 07:43:32 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
Tell calling method to send the message to sender
|
|
|
|
*/
|
2017-08-14 01:06:12 +02:00
|
|
|
if (!broadcast_line)
|
2015-07-25 07:43:32 +02:00
|
|
|
return line;
|
|
|
|
|
2017-08-14 01:06:12 +02:00
|
|
|
/*
|
|
|
|
Send the message to others
|
|
|
|
*/
|
|
|
|
actionstream << "CHAT: " << wide_to_narrow(unescape_enriched(line)) << std::endl;
|
2015-07-25 07:43:32 +02:00
|
|
|
|
2017-09-27 23:48:06 +02:00
|
|
|
std::vector<session_t> clients = m_clients.getClientIDs();
|
2017-01-17 00:09:47 +01:00
|
|
|
|
2017-08-14 01:06:12 +02:00
|
|
|
/*
|
|
|
|
Send the message back to the inital sender
|
|
|
|
if they are using protocol version >= 29
|
|
|
|
*/
|
2017-01-17 00:09:47 +01:00
|
|
|
|
2017-09-30 12:00:05 +02:00
|
|
|
session_t peer_id_to_avoid_sending =
|
|
|
|
(player ? player->getPeerId() : PEER_ID_INEXISTENT);
|
|
|
|
|
2017-08-14 01:06:12 +02:00
|
|
|
if (player && player->protocol_version >= 29)
|
|
|
|
peer_id_to_avoid_sending = PEER_ID_INEXISTENT;
|
|
|
|
|
2017-08-15 09:30:31 +02:00
|
|
|
for (u16 cid : clients) {
|
2017-08-14 01:06:12 +02:00
|
|
|
if (cid != peer_id_to_avoid_sending)
|
|
|
|
SendChatMessage(cid, ChatMessage(line));
|
2015-07-25 07:43:32 +02:00
|
|
|
}
|
|
|
|
return L"";
|
|
|
|
}
|
|
|
|
|
|
|
|
void Server::handleAdminChat(const ChatEventChat *evt)
|
|
|
|
{
|
|
|
|
std::string name = evt->nick;
|
|
|
|
std::wstring wname = utf8_to_wide(name);
|
|
|
|
std::wstring wmessage = evt->evt_msg;
|
|
|
|
|
|
|
|
std::wstring answer = handleChat(name, wname, wmessage);
|
|
|
|
|
|
|
|
// If asked to send answer to sender
|
|
|
|
if (!answer.empty()) {
|
|
|
|
m_admin_chat->outgoing_queue.push_back(new ChatEventChat("", answer));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-09-27 19:47:36 +02:00
|
|
|
RemoteClient *Server::getClient(session_t peer_id, ClientState state_min)
|
2010-11-27 00:02:21 +01:00
|
|
|
{
|
2014-01-31 00:24:00 +01:00
|
|
|
RemoteClient *client = getClientNoEx(peer_id,state_min);
|
2013-08-04 07:17:07 +02:00
|
|
|
if(!client)
|
|
|
|
throw ClientNotFoundException("Client not found");
|
2014-01-31 00:24:00 +01:00
|
|
|
|
2013-08-04 07:17:07 +02:00
|
|
|
return client;
|
|
|
|
}
|
2017-09-27 19:47:36 +02:00
|
|
|
RemoteClient *Server::getClientNoEx(session_t peer_id, ClientState state_min)
|
2013-08-04 07:17:07 +02:00
|
|
|
{
|
2014-01-31 00:24:00 +01:00
|
|
|
return m_clients.getClientNoEx(peer_id, state_min);
|
2010-11-27 00:02:21 +01:00
|
|
|
}
|
|
|
|
|
2017-09-27 19:47:36 +02:00
|
|
|
std::string Server::getPlayerName(session_t peer_id)
|
2013-08-11 04:09:45 +02:00
|
|
|
{
|
2016-10-08 16:31:22 +02:00
|
|
|
RemotePlayer *player = m_env->getPlayer(peer_id);
|
2017-08-15 09:30:31 +02:00
|
|
|
if (!player)
|
2013-08-11 04:09:45 +02:00
|
|
|
return "[id="+itos(peer_id)+"]";
|
|
|
|
return player->getName();
|
|
|
|
}
|
|
|
|
|
2017-09-27 19:47:36 +02:00
|
|
|
PlayerSAO *Server::getPlayerSAO(session_t peer_id)
|
2013-08-11 04:09:45 +02:00
|
|
|
{
|
2016-10-08 12:21:41 +02:00
|
|
|
RemotePlayer *player = m_env->getPlayer(peer_id);
|
2017-08-15 09:30:31 +02:00
|
|
|
if (!player)
|
2013-08-11 04:09:45 +02:00
|
|
|
return NULL;
|
|
|
|
return player->getPlayerSAO();
|
|
|
|
}
|
|
|
|
|
2011-02-15 15:11:24 +01:00
|
|
|
std::wstring Server::getStatusString()
|
|
|
|
{
|
|
|
|
std::wostringstream os(std::ios_base::binary);
|
2018-11-12 15:34:01 +01:00
|
|
|
os << L"# Server: ";
|
2011-04-24 23:24:40 +02:00
|
|
|
// Version
|
2018-11-12 15:34:01 +01:00
|
|
|
os << L"version=" << narrow_to_wide(g_version_string);
|
2011-02-15 15:11:24 +01:00
|
|
|
// Uptime
|
2018-11-12 15:34:01 +01:00
|
|
|
os << L", uptime=" << m_uptime.get();
|
2013-08-03 22:16:37 +02:00
|
|
|
// Max lag estimate
|
2018-11-12 15:34:01 +01:00
|
|
|
os << L", max_lag=" << (m_env ? m_env->getMaxLagEstimate() : 0);
|
|
|
|
|
2011-02-15 15:11:24 +01:00
|
|
|
// Information about clients
|
2014-01-31 00:24:00 +01:00
|
|
|
bool first = true;
|
2018-11-12 15:34:01 +01:00
|
|
|
os << L", clients={";
|
|
|
|
if (m_env) {
|
|
|
|
std::vector<session_t> clients = m_clients.getClientIDs();
|
|
|
|
for (session_t client_id : clients) {
|
|
|
|
RemotePlayer *player = m_env->getPlayer(client_id);
|
|
|
|
|
|
|
|
// Get name of player
|
|
|
|
std::wstring name = L"unknown";
|
|
|
|
if (player)
|
|
|
|
name = narrow_to_wide(player->getName());
|
|
|
|
|
|
|
|
// Add name to information string
|
|
|
|
if (!first)
|
|
|
|
os << L", ";
|
|
|
|
else
|
|
|
|
first = false;
|
|
|
|
|
|
|
|
os << name;
|
|
|
|
}
|
2011-02-15 15:11:24 +01:00
|
|
|
}
|
2015-03-04 16:30:24 +01:00
|
|
|
os << L"}";
|
2017-08-14 01:06:12 +02:00
|
|
|
|
2018-11-12 15:34:01 +01:00
|
|
|
if (m_env && !((ServerMap*)(&m_env->getMap()))->isSavingEnabled())
|
|
|
|
os << std::endl << L"# Server: " << " WARNING: Map saving is disabled.";
|
2017-08-14 01:06:12 +02:00
|
|
|
|
|
|
|
if (!g_settings->get("motd").empty())
|
2018-11-12 15:34:01 +01:00
|
|
|
os << std::endl << L"# Server: " << narrow_to_wide(g_settings->get("motd"));
|
|
|
|
|
2011-02-15 15:11:24 +01:00
|
|
|
return os.str();
|
|
|
|
}
|
|
|
|
|
2012-03-30 17:42:18 +02:00
|
|
|
std::set<std::string> Server::getPlayerEffectivePrivs(const std::string &name)
|
2012-03-15 14:27:16 +01:00
|
|
|
{
|
2012-03-30 17:42:18 +02:00
|
|
|
std::set<std::string> privs;
|
2013-05-25 00:51:02 +02:00
|
|
|
m_script->getAuth(name, NULL, &privs);
|
2012-03-30 17:42:18 +02:00
|
|
|
return privs;
|
2012-03-15 14:27:16 +01:00
|
|
|
}
|
|
|
|
|
2012-03-30 17:42:18 +02:00
|
|
|
bool Server::checkPriv(const std::string &name, const std::string &priv)
|
2011-11-20 20:16:15 +01:00
|
|
|
{
|
2012-03-30 17:42:18 +02:00
|
|
|
std::set<std::string> privs = getPlayerEffectivePrivs(name);
|
|
|
|
return (privs.count(priv) != 0);
|
2011-11-20 20:16:15 +01:00
|
|
|
}
|
|
|
|
|
2012-03-31 15:23:26 +02:00
|
|
|
void Server::reportPrivsModified(const std::string &name)
|
|
|
|
{
|
2017-08-14 01:06:12 +02:00
|
|
|
if (name.empty()) {
|
2017-09-27 23:48:06 +02:00
|
|
|
std::vector<session_t> clients = m_clients.getClientIDs();
|
|
|
|
for (const session_t client_id : clients) {
|
2017-08-14 01:06:12 +02:00
|
|
|
RemotePlayer *player = m_env->getPlayer(client_id);
|
2012-03-31 16:08:39 +02:00
|
|
|
reportPrivsModified(player->getName());
|
2012-03-31 15:23:26 +02:00
|
|
|
}
|
|
|
|
} else {
|
2016-10-08 16:31:22 +02:00
|
|
|
RemotePlayer *player = m_env->getPlayer(name.c_str());
|
2016-10-08 10:38:04 +02:00
|
|
|
if (!player)
|
2012-03-31 15:23:26 +02:00
|
|
|
return;
|
2017-09-30 12:00:05 +02:00
|
|
|
SendPlayerPrivileges(player->getPeerId());
|
2012-04-07 17:46:10 +02:00
|
|
|
PlayerSAO *sao = player->getPlayerSAO();
|
|
|
|
if(!sao)
|
|
|
|
return;
|
|
|
|
sao->updatePrivileges(
|
2012-04-01 12:19:50 +02:00
|
|
|
getPlayerEffectivePrivs(name),
|
|
|
|
isSingleplayer());
|
2012-03-31 15:23:26 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-07-19 13:09:16 +02:00
|
|
|
void Server::reportInventoryFormspecModified(const std::string &name)
|
|
|
|
{
|
2016-10-08 16:31:22 +02:00
|
|
|
RemotePlayer *player = m_env->getPlayer(name.c_str());
|
|
|
|
if (!player)
|
2012-07-19 13:09:16 +02:00
|
|
|
return;
|
2017-09-30 12:00:05 +02:00
|
|
|
SendPlayerInventoryFormspec(player->getPeerId());
|
2012-07-19 13:09:16 +02:00
|
|
|
}
|
|
|
|
|
2018-03-28 17:04:41 +02:00
|
|
|
void Server::reportFormspecPrependModified(const std::string &name)
|
|
|
|
{
|
|
|
|
RemotePlayer *player = m_env->getPlayer(name.c_str());
|
|
|
|
if (!player)
|
|
|
|
return;
|
|
|
|
SendPlayerFormspecPrepend(player->getPeerId());
|
|
|
|
}
|
|
|
|
|
2013-08-11 04:09:45 +02:00
|
|
|
void Server::setIpBanned(const std::string &ip, const std::string &name)
|
2011-10-12 12:53:38 +02:00
|
|
|
{
|
2013-08-11 04:09:45 +02:00
|
|
|
m_banmanager->add(ip, name);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Server::unsetIpBanned(const std::string &ip_or_name)
|
|
|
|
{
|
|
|
|
m_banmanager->remove(ip_or_name);
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string Server::getBanDescription(const std::string &ip_or_name)
|
|
|
|
{
|
|
|
|
return m_banmanager->getBanDescription(ip_or_name);
|
2011-10-12 12:53:38 +02:00
|
|
|
}
|
|
|
|
|
2014-02-27 21:12:59 +01:00
|
|
|
void Server::notifyPlayer(const char *name, const std::wstring &msg)
|
2011-10-16 19:03:43 +02:00
|
|
|
{
|
2015-08-09 22:10:37 +02:00
|
|
|
// m_env will be NULL if the server is initializing
|
|
|
|
if (!m_env)
|
|
|
|
return;
|
|
|
|
|
2015-07-25 07:43:32 +02:00
|
|
|
if (m_admin_nick == name && !m_admin_nick.empty()) {
|
|
|
|
m_admin_chat->outgoing_queue.push_back(new ChatEventChat("", msg));
|
|
|
|
}
|
|
|
|
|
2016-10-08 16:31:22 +02:00
|
|
|
RemotePlayer *player = m_env->getPlayer(name);
|
2015-07-25 07:43:32 +02:00
|
|
|
if (!player) {
|
2011-10-16 19:03:43 +02:00
|
|
|
return;
|
2015-07-25 07:43:32 +02:00
|
|
|
}
|
2014-02-02 01:55:24 +01:00
|
|
|
|
2017-09-30 12:00:05 +02:00
|
|
|
if (player->getPeerId() == PEER_ID_INEXISTENT)
|
2014-02-02 01:55:24 +01:00
|
|
|
return;
|
|
|
|
|
2017-09-30 12:00:05 +02:00
|
|
|
SendChatMessage(player->getPeerId(), ChatMessage(msg));
|
2011-10-16 19:03:43 +02:00
|
|
|
}
|
|
|
|
|
2015-07-03 05:14:30 +02:00
|
|
|
bool Server::showFormspec(const char *playername, const std::string &formspec,
|
|
|
|
const std::string &formname)
|
2013-01-02 20:45:04 +01:00
|
|
|
{
|
2015-08-09 22:10:37 +02:00
|
|
|
// m_env will be NULL if the server is initializing
|
|
|
|
if (!m_env)
|
|
|
|
return false;
|
|
|
|
|
2016-10-08 16:31:22 +02:00
|
|
|
RemotePlayer *player = m_env->getPlayer(playername);
|
2015-07-03 05:14:30 +02:00
|
|
|
if (!player)
|
2013-01-02 20:45:04 +01:00
|
|
|
return false;
|
|
|
|
|
2017-09-30 12:00:05 +02:00
|
|
|
SendShowFormspecMessage(player->getPeerId(), formspec, formname);
|
2013-01-02 20:45:04 +01:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2016-10-08 12:21:41 +02:00
|
|
|
u32 Server::hudAdd(RemotePlayer *player, HudElement *form)
|
2015-07-03 05:14:30 +02:00
|
|
|
{
|
2013-04-14 00:20:22 +02:00
|
|
|
if (!player)
|
|
|
|
return -1;
|
2014-12-17 09:20:17 +01:00
|
|
|
|
2014-05-25 14:34:32 +02:00
|
|
|
u32 id = player->addHud(form);
|
|
|
|
|
2017-09-30 12:00:05 +02:00
|
|
|
SendHUDAdd(player->getPeerId(), id, form);
|
2014-05-25 14:34:32 +02:00
|
|
|
|
2013-04-14 00:20:22 +02:00
|
|
|
return id;
|
2013-04-11 20:23:38 +02:00
|
|
|
}
|
|
|
|
|
2016-10-08 12:21:41 +02:00
|
|
|
bool Server::hudRemove(RemotePlayer *player, u32 id) {
|
2014-05-25 14:34:32 +02:00
|
|
|
if (!player)
|
2013-04-11 20:23:38 +02:00
|
|
|
return false;
|
|
|
|
|
2014-05-25 14:34:32 +02:00
|
|
|
HudElement* todel = player->removeHud(id);
|
|
|
|
|
|
|
|
if (!todel)
|
|
|
|
return false;
|
2014-12-17 09:20:17 +01:00
|
|
|
|
2014-05-25 14:34:32 +02:00
|
|
|
delete todel;
|
|
|
|
|
2017-09-30 12:00:05 +02:00
|
|
|
SendHUDRemove(player->getPeerId(), id);
|
2013-04-11 20:23:38 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2016-10-08 12:21:41 +02:00
|
|
|
bool Server::hudChange(RemotePlayer *player, u32 id, HudElementStat stat, void *data)
|
2015-07-03 05:14:30 +02:00
|
|
|
{
|
2013-04-14 00:20:22 +02:00
|
|
|
if (!player)
|
2013-04-11 20:23:38 +02:00
|
|
|
return false;
|
|
|
|
|
2017-09-30 12:00:05 +02:00
|
|
|
SendHUDChange(player->getPeerId(), id, stat, data);
|
2013-04-11 20:23:38 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2016-10-08 10:38:04 +02:00
|
|
|
bool Server::hudSetFlags(RemotePlayer *player, u32 flags, u32 mask)
|
2015-07-03 05:14:30 +02:00
|
|
|
{
|
2013-04-24 12:52:46 +02:00
|
|
|
if (!player)
|
|
|
|
return false;
|
|
|
|
|
2017-09-30 12:00:05 +02:00
|
|
|
SendHUDSetFlags(player->getPeerId(), flags, mask);
|
2015-11-22 16:15:47 +01:00
|
|
|
player->hud_flags &= ~mask;
|
|
|
|
player->hud_flags |= flags;
|
2014-12-17 09:20:17 +01:00
|
|
|
|
2014-08-21 17:27:52 +02:00
|
|
|
PlayerSAO* playersao = player->getPlayerSAO();
|
2014-12-17 09:20:17 +01:00
|
|
|
|
2017-08-14 01:06:12 +02:00
|
|
|
if (!playersao)
|
2014-08-21 17:27:52 +02:00
|
|
|
return false;
|
2014-04-28 23:41:27 +02:00
|
|
|
|
2014-08-21 17:27:52 +02:00
|
|
|
m_script->player_event(playersao, "hud_changed");
|
2013-04-24 12:52:46 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2016-10-08 10:38:04 +02:00
|
|
|
bool Server::hudSetHotbarItemcount(RemotePlayer *player, s32 hotbar_itemcount)
|
2015-07-03 05:14:30 +02:00
|
|
|
{
|
2013-05-04 02:08:52 +02:00
|
|
|
if (!player)
|
|
|
|
return false;
|
2016-10-08 10:38:04 +02:00
|
|
|
|
2013-05-04 02:08:52 +02:00
|
|
|
if (hotbar_itemcount <= 0 || hotbar_itemcount > HUD_HOTBAR_ITEMCOUNT_MAX)
|
|
|
|
return false;
|
|
|
|
|
2015-05-26 14:10:08 +02:00
|
|
|
player->setHotbarItemcount(hotbar_itemcount);
|
2013-05-04 02:08:52 +02:00
|
|
|
std::ostringstream os(std::ios::binary);
|
|
|
|
writeS32(os, hotbar_itemcount);
|
2017-09-30 12:00:05 +02:00
|
|
|
SendHUDSetParam(player->getPeerId(), HUD_PARAM_HOTBAR_ITEMCOUNT, os.str());
|
2013-05-04 02:08:52 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2016-10-08 12:21:41 +02:00
|
|
|
void Server::hudSetHotbarImage(RemotePlayer *player, std::string name)
|
2015-07-03 05:14:30 +02:00
|
|
|
{
|
2013-09-03 19:51:40 +02:00
|
|
|
if (!player)
|
|
|
|
return;
|
|
|
|
|
2015-05-26 14:10:08 +02:00
|
|
|
player->setHotbarImage(name);
|
2017-09-30 12:00:05 +02:00
|
|
|
SendHUDSetParam(player->getPeerId(), HUD_PARAM_HOTBAR_IMAGE, name);
|
2013-09-03 19:51:40 +02:00
|
|
|
}
|
|
|
|
|
2016-10-08 12:21:41 +02:00
|
|
|
void Server::hudSetHotbarSelectedImage(RemotePlayer *player, std::string name)
|
2015-07-03 05:14:30 +02:00
|
|
|
{
|
2013-09-03 19:51:40 +02:00
|
|
|
if (!player)
|
|
|
|
return;
|
|
|
|
|
2015-05-26 14:10:08 +02:00
|
|
|
player->setHotbarSelectedImage(name);
|
2017-09-30 12:00:05 +02:00
|
|
|
SendHUDSetParam(player->getPeerId(), HUD_PARAM_HOTBAR_SELECTED_IMAGE, name);
|
2013-09-03 19:51:40 +02:00
|
|
|
}
|
|
|
|
|
2017-09-27 19:47:36 +02:00
|
|
|
Address Server::getPeerAddress(session_t peer_id)
|
2017-08-24 08:28:54 +02:00
|
|
|
{
|
|
|
|
return m_con->GetPeerAddress(peer_id);
|
|
|
|
}
|
|
|
|
|
2018-03-08 22:19:25 +01:00
|
|
|
void Server::setLocalPlayerAnimations(RemotePlayer *player,
|
2016-10-08 12:21:41 +02:00
|
|
|
v2s32 animation_frames[4], f32 frame_speed)
|
2014-01-08 13:47:53 +01:00
|
|
|
{
|
2018-03-08 22:19:25 +01:00
|
|
|
sanity_check(player);
|
2015-05-26 14:10:08 +02:00
|
|
|
player->setLocalAnimations(animation_frames, frame_speed);
|
2017-09-30 12:00:05 +02:00
|
|
|
SendLocalPlayerAnimations(player->getPeerId(), animation_frames, frame_speed);
|
2014-01-08 13:47:53 +01:00
|
|
|
}
|
|
|
|
|
2018-03-08 22:19:25 +01:00
|
|
|
void Server::setPlayerEyeOffset(RemotePlayer *player, const v3f &first, const v3f &third)
|
2014-04-11 15:32:46 +02:00
|
|
|
{
|
2018-03-08 22:19:25 +01:00
|
|
|
sanity_check(player);
|
2015-05-26 14:10:08 +02:00
|
|
|
player->eye_offset_first = first;
|
|
|
|
player->eye_offset_third = third;
|
2017-09-30 12:00:05 +02:00
|
|
|
SendEyeOffset(player->getPeerId(), first, third);
|
2014-04-11 15:32:46 +02:00
|
|
|
}
|
|
|
|
|
2018-03-08 22:19:25 +01:00
|
|
|
void Server::setSky(RemotePlayer *player, const video::SColor &bgcolor,
|
2017-04-28 04:06:49 +02:00
|
|
|
const std::string &type, const std::vector<std::string> ¶ms,
|
|
|
|
bool &clouds)
|
2013-05-02 22:52:50 +02:00
|
|
|
{
|
2018-03-08 22:19:25 +01:00
|
|
|
sanity_check(player);
|
2017-04-28 04:06:49 +02:00
|
|
|
player->setSky(bgcolor, type, params, clouds);
|
2017-09-30 12:00:05 +02:00
|
|
|
SendSetSky(player->getPeerId(), bgcolor, type, params, clouds);
|
2013-05-02 22:52:50 +02:00
|
|
|
}
|
|
|
|
|
2018-03-08 22:19:25 +01:00
|
|
|
void Server::setClouds(RemotePlayer *player, const CloudParams ¶ms)
|
2017-03-17 10:39:47 +01:00
|
|
|
{
|
2018-03-08 22:19:25 +01:00
|
|
|
sanity_check(player);
|
|
|
|
player->setCloudParams(params);
|
|
|
|
SendCloudParams(player->getPeerId(), params);
|
2017-03-17 10:39:47 +01:00
|
|
|
}
|
|
|
|
|
2016-10-08 10:38:04 +02:00
|
|
|
bool Server::overrideDayNightRatio(RemotePlayer *player, bool do_override,
|
2015-07-03 05:14:30 +02:00
|
|
|
float ratio)
|
2013-10-18 10:53:19 +02:00
|
|
|
{
|
|
|
|
if (!player)
|
|
|
|
return false;
|
|
|
|
|
2015-05-26 14:10:08 +02:00
|
|
|
player->overrideDayNightRatio(do_override, ratio);
|
2017-09-30 12:00:05 +02:00
|
|
|
SendOverrideDayNightRatio(player->getPeerId(), do_override, ratio);
|
2013-10-18 10:53:19 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2014-02-27 21:12:59 +01:00
|
|
|
void Server::notifyPlayers(const std::wstring &msg)
|
2011-10-17 23:01:50 +02:00
|
|
|
{
|
2017-07-16 10:47:31 +02:00
|
|
|
SendChatMessage(PEER_ID_INEXISTENT, ChatMessage(msg));
|
2011-10-17 23:01:50 +02:00
|
|
|
}
|
|
|
|
|
2015-07-25 11:56:24 +02:00
|
|
|
void Server::spawnParticle(const std::string &playername, v3f pos,
|
2015-07-03 05:14:30 +02:00
|
|
|
v3f velocity, v3f acceleration,
|
|
|
|
float expirationtime, float size, bool
|
2018-09-08 01:38:35 +02:00
|
|
|
collisiondetection, bool collision_removal, bool object_collision,
|
2017-01-14 16:48:49 +01:00
|
|
|
bool vertical, const std::string &texture,
|
|
|
|
const struct TileAnimationParams &animation, u8 glow)
|
2013-01-23 18:32:02 +01:00
|
|
|
{
|
2015-08-09 22:10:37 +02:00
|
|
|
// m_env will be NULL if the server is initializing
|
|
|
|
if (!m_env)
|
|
|
|
return;
|
|
|
|
|
2017-09-27 19:47:36 +02:00
|
|
|
session_t peer_id = PEER_ID_INEXISTENT;
|
|
|
|
u16 proto_ver = 0;
|
2017-08-14 01:06:12 +02:00
|
|
|
if (!playername.empty()) {
|
2016-10-08 19:08:23 +02:00
|
|
|
RemotePlayer *player = m_env->getPlayer(playername.c_str());
|
2015-07-25 11:56:24 +02:00
|
|
|
if (!player)
|
|
|
|
return;
|
2017-09-30 12:00:05 +02:00
|
|
|
peer_id = player->getPeerId();
|
2017-01-14 16:48:49 +01:00
|
|
|
proto_ver = player->protocol_version;
|
2015-07-25 11:56:24 +02:00
|
|
|
}
|
2013-01-23 18:32:02 +01:00
|
|
|
|
2017-01-14 16:48:49 +01:00
|
|
|
SendSpawnParticle(peer_id, proto_ver, pos, velocity, acceleration,
|
2018-09-08 01:38:35 +02:00
|
|
|
expirationtime, size, collisiondetection, collision_removal,
|
|
|
|
object_collision, vertical, texture, animation, glow);
|
2013-01-23 18:32:02 +01:00
|
|
|
}
|
|
|
|
|
2015-07-25 11:56:24 +02:00
|
|
|
u32 Server::addParticleSpawner(u16 amount, float spawntime,
|
2015-07-03 05:14:30 +02:00
|
|
|
v3f minpos, v3f maxpos, v3f minvel, v3f maxvel, v3f minacc, v3f maxacc,
|
|
|
|
float minexptime, float maxexptime, float minsize, float maxsize,
|
2018-09-08 01:38:35 +02:00
|
|
|
bool collisiondetection, bool collision_removal, bool object_collision,
|
2016-08-04 22:09:21 +02:00
|
|
|
ServerActiveObject *attached, bool vertical, const std::string &texture,
|
2017-01-14 16:48:49 +01:00
|
|
|
const std::string &playername, const struct TileAnimationParams &animation,
|
|
|
|
u8 glow)
|
2013-01-23 18:32:02 +01:00
|
|
|
{
|
2015-08-09 22:10:37 +02:00
|
|
|
// m_env will be NULL if the server is initializing
|
|
|
|
if (!m_env)
|
|
|
|
return -1;
|
|
|
|
|
2017-09-27 19:47:36 +02:00
|
|
|
session_t peer_id = PEER_ID_INEXISTENT;
|
|
|
|
u16 proto_ver = 0;
|
2017-08-14 01:06:12 +02:00
|
|
|
if (!playername.empty()) {
|
2016-10-08 19:08:23 +02:00
|
|
|
RemotePlayer *player = m_env->getPlayer(playername.c_str());
|
2015-07-25 11:56:24 +02:00
|
|
|
if (!player)
|
|
|
|
return -1;
|
2017-09-30 12:00:05 +02:00
|
|
|
peer_id = player->getPeerId();
|
2017-01-14 16:48:49 +01:00
|
|
|
proto_ver = player->protocol_version;
|
2013-01-23 18:32:02 +01:00
|
|
|
}
|
|
|
|
|
2016-08-04 22:09:21 +02:00
|
|
|
u16 attached_id = attached ? attached->getId() : 0;
|
|
|
|
|
|
|
|
u32 id;
|
|
|
|
if (attached_id == 0)
|
|
|
|
id = m_env->addParticleSpawner(spawntime);
|
|
|
|
else
|
|
|
|
id = m_env->addParticleSpawner(spawntime, attached_id);
|
|
|
|
|
2017-01-14 16:48:49 +01:00
|
|
|
SendAddParticleSpawner(peer_id, proto_ver, amount, spawntime,
|
2013-01-23 18:32:02 +01:00
|
|
|
minpos, maxpos, minvel, maxvel, minacc, maxacc,
|
2018-09-08 01:38:35 +02:00
|
|
|
minexptime, maxexptime, minsize, maxsize, collisiondetection,
|
|
|
|
collision_removal, object_collision, attached_id, vertical,
|
2017-01-14 16:48:49 +01:00
|
|
|
texture, id, animation, glow);
|
2013-01-23 18:32:02 +01:00
|
|
|
|
|
|
|
return id;
|
|
|
|
}
|
|
|
|
|
2015-07-25 11:56:24 +02:00
|
|
|
void Server::deleteParticleSpawner(const std::string &playername, u32 id)
|
2013-01-23 18:32:02 +01:00
|
|
|
{
|
2015-08-09 22:10:37 +02:00
|
|
|
// m_env will be NULL if the server is initializing
|
|
|
|
if (!m_env)
|
|
|
|
throw ServerError("Can't delete particle spawners during initialisation!");
|
|
|
|
|
2017-09-27 19:47:36 +02:00
|
|
|
session_t peer_id = PEER_ID_INEXISTENT;
|
2017-08-14 01:06:12 +02:00
|
|
|
if (!playername.empty()) {
|
2016-10-08 19:08:23 +02:00
|
|
|
RemotePlayer *player = m_env->getPlayer(playername.c_str());
|
2015-07-25 11:56:24 +02:00
|
|
|
if (!player)
|
|
|
|
return;
|
2017-09-30 12:00:05 +02:00
|
|
|
peer_id = player->getPeerId();
|
2015-07-25 11:56:24 +02:00
|
|
|
}
|
2013-01-23 18:32:02 +01:00
|
|
|
|
2016-04-27 17:58:09 +02:00
|
|
|
m_env->deleteParticleSpawner(id);
|
2015-07-25 11:56:24 +02:00
|
|
|
SendDeleteParticleSpawner(peer_id, id);
|
2013-01-23 18:32:02 +01:00
|
|
|
}
|
|
|
|
|
2016-11-26 17:35:25 +01:00
|
|
|
Inventory* Server::createDetachedInventory(const std::string &name, const std::string &player)
|
2012-07-24 19:57:17 +02:00
|
|
|
{
|
|
|
|
if(m_detached_inventories.count(name) > 0){
|
|
|
|
infostream<<"Server clearing detached inventory \""<<name<<"\""<<std::endl;
|
|
|
|
delete m_detached_inventories[name];
|
|
|
|
} else {
|
|
|
|
infostream<<"Server creating detached inventory \""<<name<<"\""<<std::endl;
|
|
|
|
}
|
|
|
|
Inventory *inv = new Inventory(m_itemdef);
|
2015-03-06 11:21:51 +01:00
|
|
|
sanity_check(inv);
|
2012-07-24 19:57:17 +02:00
|
|
|
m_detached_inventories[name] = inv;
|
2016-11-26 17:35:25 +01:00
|
|
|
m_detached_inventories_player[name] = player;
|
2014-01-31 00:24:00 +01:00
|
|
|
//TODO find a better way to do this
|
|
|
|
sendDetachedInventory(name,PEER_ID_INEXISTENT);
|
2012-07-24 19:57:17 +02:00
|
|
|
return inv;
|
|
|
|
}
|
|
|
|
|
2018-10-10 20:48:58 +02:00
|
|
|
bool Server::removeDetachedInventory(const std::string &name)
|
|
|
|
{
|
|
|
|
const auto &inv_it = m_detached_inventories.find(name);
|
|
|
|
if (inv_it == m_detached_inventories.end())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
delete inv_it->second;
|
|
|
|
m_detached_inventories.erase(inv_it);
|
|
|
|
|
|
|
|
const auto &player_it = m_detached_inventories_player.find(name);
|
|
|
|
if (player_it != m_detached_inventories_player.end()) {
|
|
|
|
RemotePlayer *player = m_env->getPlayer(player_it->second.c_str());
|
|
|
|
|
|
|
|
if (player && player->getPeerId() != PEER_ID_INEXISTENT)
|
|
|
|
sendDetachedInventory(name, player->getPeerId());
|
|
|
|
|
|
|
|
m_detached_inventories_player.erase(player_it);
|
|
|
|
} else {
|
|
|
|
// Notify all players about the change
|
|
|
|
sendDetachedInventory(name, PEER_ID_INEXISTENT);
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2012-07-26 21:06:45 +02:00
|
|
|
// actions: time-reversed list
|
|
|
|
// Return value: success/failure
|
|
|
|
bool Server::rollbackRevertActions(const std::list<RollbackAction> &actions,
|
|
|
|
std::list<std::string> *log)
|
|
|
|
{
|
|
|
|
infostream<<"Server::rollbackRevertActions(len="<<actions.size()<<")"<<std::endl;
|
|
|
|
ServerMap *map = (ServerMap*)(&m_env->getMap());
|
2012-11-26 03:16:48 +01:00
|
|
|
|
2012-07-26 21:06:45 +02:00
|
|
|
// Fail if no actions to handle
|
2018-04-04 08:40:21 +02:00
|
|
|
if (actions.empty()) {
|
|
|
|
assert(log);
|
2012-07-26 21:06:45 +02:00
|
|
|
log->push_back("Nothing to do.");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
int num_tried = 0;
|
|
|
|
int num_failed = 0;
|
2012-11-26 03:16:48 +01:00
|
|
|
|
2017-08-14 01:06:12 +02:00
|
|
|
for (const RollbackAction &action : actions) {
|
2012-07-26 21:06:45 +02:00
|
|
|
num_tried++;
|
|
|
|
bool success = action.applyRevert(map, this, this);
|
|
|
|
if(!success){
|
|
|
|
num_failed++;
|
|
|
|
std::ostringstream os;
|
|
|
|
os<<"Revert of step ("<<num_tried<<") "<<action.toString()<<" failed";
|
|
|
|
infostream<<"Map::rollbackRevertActions(): "<<os.str()<<std::endl;
|
2017-08-14 01:06:12 +02:00
|
|
|
if (log)
|
2012-07-26 21:06:45 +02:00
|
|
|
log->push_back(os.str());
|
|
|
|
}else{
|
|
|
|
std::ostringstream os;
|
2012-12-17 16:46:54 +01:00
|
|
|
os<<"Successfully reverted step ("<<num_tried<<") "<<action.toString();
|
2012-07-26 21:06:45 +02:00
|
|
|
infostream<<"Map::rollbackRevertActions(): "<<os.str()<<std::endl;
|
2017-08-14 01:06:12 +02:00
|
|
|
if (log)
|
2012-07-26 21:06:45 +02:00
|
|
|
log->push_back(os.str());
|
|
|
|
}
|
|
|
|
}
|
2012-11-26 03:16:48 +01:00
|
|
|
|
2012-07-26 21:06:45 +02:00
|
|
|
infostream<<"Map::rollbackRevertActions(): "<<num_failed<<"/"<<num_tried
|
|
|
|
<<" failed"<<std::endl;
|
|
|
|
|
|
|
|
// Call it done if less than half failed
|
|
|
|
return num_failed <= num_tried/2;
|
|
|
|
}
|
|
|
|
|
2011-11-14 20:41:30 +01:00
|
|
|
// IGameDef interface
|
|
|
|
// Under envlock
|
2015-07-03 05:14:30 +02:00
|
|
|
IItemDefManager *Server::getItemDefManager()
|
2011-11-14 20:41:30 +01:00
|
|
|
{
|
2012-01-12 06:10:39 +01:00
|
|
|
return m_itemdef;
|
2011-11-14 20:41:30 +01:00
|
|
|
}
|
2015-07-03 05:14:30 +02:00
|
|
|
|
2018-02-10 21:04:16 +01:00
|
|
|
const NodeDefManager *Server::getNodeDefManager()
|
2011-11-14 20:41:30 +01:00
|
|
|
{
|
2011-11-16 13:08:31 +01:00
|
|
|
return m_nodedef;
|
2011-11-14 20:41:30 +01:00
|
|
|
}
|
2015-07-03 05:14:30 +02:00
|
|
|
|
|
|
|
ICraftDefManager *Server::getCraftDefManager()
|
2011-11-17 01:28:46 +01:00
|
|
|
{
|
|
|
|
return m_craftdef;
|
|
|
|
}
|
2014-10-21 05:12:15 +02:00
|
|
|
|
2011-11-16 12:03:28 +01:00
|
|
|
u16 Server::allocateUnknownNodeId(const std::string &name)
|
|
|
|
{
|
2011-11-16 13:08:31 +01:00
|
|
|
return m_nodedef->allocateDummy(name);
|
2011-11-16 12:03:28 +01:00
|
|
|
}
|
2015-07-03 05:14:30 +02:00
|
|
|
|
|
|
|
IWritableItemDefManager *Server::getWritableItemDefManager()
|
2011-11-15 01:03:28 +01:00
|
|
|
{
|
2012-01-12 06:10:39 +01:00
|
|
|
return m_itemdef;
|
2011-11-15 01:03:28 +01:00
|
|
|
}
|
2015-07-03 05:14:30 +02:00
|
|
|
|
2018-02-10 21:04:16 +01:00
|
|
|
NodeDefManager *Server::getWritableNodeDefManager()
|
2011-11-15 01:03:28 +01:00
|
|
|
{
|
2011-11-16 13:08:31 +01:00
|
|
|
return m_nodedef;
|
2011-11-15 01:03:28 +01:00
|
|
|
}
|
2015-07-03 05:14:30 +02:00
|
|
|
|
|
|
|
IWritableCraftDefManager *Server::getWritableCraftDefManager()
|
2011-11-17 01:28:46 +01:00
|
|
|
{
|
|
|
|
return m_craftdef;
|
|
|
|
}
|
2011-11-15 01:03:28 +01:00
|
|
|
|
2018-03-16 08:41:33 +01:00
|
|
|
const std::vector<ModSpec> & Server::getMods() const
|
|
|
|
{
|
|
|
|
return m_modmgr->getMods();
|
|
|
|
}
|
|
|
|
|
2015-07-03 05:14:30 +02:00
|
|
|
const ModSpec *Server::getModSpec(const std::string &modname) const
|
2011-12-11 15:49:40 +01:00
|
|
|
{
|
2018-03-16 08:41:33 +01:00
|
|
|
return m_modmgr->getModSpec(modname);
|
2011-12-11 15:49:40 +01:00
|
|
|
}
|
2015-07-03 05:14:30 +02:00
|
|
|
|
2015-03-05 10:43:08 +01:00
|
|
|
void Server::getModNames(std::vector<std::string> &modlist)
|
2012-07-22 03:29:37 +02:00
|
|
|
{
|
2018-03-16 08:41:33 +01:00
|
|
|
m_modmgr->getModNames(modlist);
|
2012-07-22 03:29:37 +02:00
|
|
|
}
|
2015-07-03 05:14:30 +02:00
|
|
|
|
2017-01-28 18:31:23 +01:00
|
|
|
std::string Server::getBuiltinLuaPath()
|
2012-04-01 09:08:52 +02:00
|
|
|
{
|
|
|
|
return porting::path_share + DIR_DELIM + "builtin";
|
|
|
|
}
|
2011-12-11 15:49:40 +01:00
|
|
|
|
2017-02-08 00:15:55 +01:00
|
|
|
std::string Server::getModStoragePath() const
|
|
|
|
{
|
|
|
|
return m_path_world + DIR_DELIM + "mod_storage";
|
|
|
|
}
|
|
|
|
|
2015-04-03 09:04:02 +02:00
|
|
|
v3f Server::findSpawnPos()
|
2011-04-21 18:35:17 +02:00
|
|
|
{
|
2015-04-03 09:04:02 +02:00
|
|
|
ServerMap &map = m_env->getServerMap();
|
|
|
|
v3f nodeposf;
|
|
|
|
if (g_settings->getV3FNoEx("static_spawnpoint", nodeposf)) {
|
|
|
|
return nodeposf * BS;
|
|
|
|
}
|
2011-06-25 03:25:14 +02:00
|
|
|
|
2015-04-03 09:04:02 +02:00
|
|
|
bool is_good = false;
|
2017-06-04 23:28:32 +02:00
|
|
|
// Limit spawn range to mapgen edges (determined by 'mapgen_limit')
|
|
|
|
s32 range_max = map.getMapgenParams()->getSpawnRangeMax();
|
2015-04-03 09:04:02 +02:00
|
|
|
|
2011-04-21 18:35:17 +02:00
|
|
|
// Try to find a good place a few times
|
2016-02-04 02:03:31 +01:00
|
|
|
for(s32 i = 0; i < 4000 && !is_good; i++) {
|
2017-06-04 23:28:32 +02:00
|
|
|
s32 range = MYMIN(1 + i, range_max);
|
2011-04-21 18:35:17 +02:00
|
|
|
// We're going to try to throw the player to this position
|
2013-04-21 21:39:34 +02:00
|
|
|
v2s16 nodepos2d = v2s16(
|
2017-04-23 14:35:08 +02:00
|
|
|
-range + (myrand() % (range * 2)),
|
|
|
|
-range + (myrand() % (range * 2)));
|
2013-04-21 21:39:34 +02:00
|
|
|
|
2016-02-04 02:03:31 +01:00
|
|
|
// Get spawn level at point
|
|
|
|
s16 spawn_level = m_emerge->getSpawnLevelAtPoint(nodepos2d);
|
|
|
|
// Continue if MAX_MAP_GENERATION_LIMIT was returned by
|
|
|
|
// the mapgen to signify an unsuitable spawn position
|
|
|
|
if (spawn_level == MAX_MAP_GENERATION_LIMIT)
|
2011-04-21 18:35:17 +02:00
|
|
|
continue;
|
2012-11-26 03:16:48 +01:00
|
|
|
|
2016-02-04 02:03:31 +01:00
|
|
|
v3s16 nodepos(nodepos2d.X, spawn_level, nodepos2d.Y);
|
2015-04-03 09:04:02 +02:00
|
|
|
|
2011-11-13 02:17:42 +01:00
|
|
|
s32 air_count = 0;
|
2013-04-21 21:39:34 +02:00
|
|
|
for (s32 i = 0; i < 10; i++) {
|
2011-11-13 02:17:42 +01:00
|
|
|
v3s16 blockpos = getNodeBlockPos(nodepos);
|
|
|
|
map.emergeBlock(blockpos, true);
|
2013-04-21 21:39:34 +02:00
|
|
|
content_t c = map.getNodeNoEx(nodepos).getContent();
|
|
|
|
if (c == CONTENT_AIR || c == CONTENT_IGNORE) {
|
2011-11-13 02:17:42 +01:00
|
|
|
air_count++;
|
2015-09-29 17:26:07 +02:00
|
|
|
if (air_count >= 2) {
|
|
|
|
nodeposf = intToFloat(nodepos, BS);
|
|
|
|
// Don't spawn the player outside map boundaries
|
|
|
|
if (objectpos_over_limit(nodeposf))
|
|
|
|
continue;
|
2011-11-13 02:17:42 +01:00
|
|
|
is_good = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
nodepos.Y++;
|
|
|
|
}
|
2011-04-21 18:35:17 +02:00
|
|
|
}
|
2012-11-26 03:16:48 +01:00
|
|
|
|
2015-09-29 17:26:07 +02:00
|
|
|
return nodeposf;
|
2011-04-21 18:35:17 +02:00
|
|
|
}
|
|
|
|
|
2017-04-15 23:19:18 +02:00
|
|
|
void Server::requestShutdown(const std::string &msg, bool reconnect, float delay)
|
|
|
|
{
|
|
|
|
if (delay == 0.0f) {
|
|
|
|
// No delay, shutdown immediately
|
2018-06-13 21:58:34 +02:00
|
|
|
m_shutdown_state.is_requested = true;
|
2017-06-04 21:00:04 +02:00
|
|
|
// only print to the infostream, a chat message saying
|
2017-04-21 19:31:59 +02:00
|
|
|
// "Server Shutting Down" is sent when the server destructs.
|
|
|
|
infostream << "*** Immediate Server shutdown requested." << std::endl;
|
2018-06-13 21:58:34 +02:00
|
|
|
} else if (delay < 0.0f && m_shutdown_state.isTimerRunning()) {
|
|
|
|
// Negative delay, cancel shutdown if requested
|
|
|
|
m_shutdown_state.reset();
|
2017-04-15 23:19:18 +02:00
|
|
|
std::wstringstream ws;
|
|
|
|
|
|
|
|
ws << L"*** Server shutdown canceled.";
|
|
|
|
|
|
|
|
infostream << wide_to_utf8(ws.str()).c_str() << std::endl;
|
|
|
|
SendChatMessage(PEER_ID_INEXISTENT, ws.str());
|
2018-06-11 13:43:12 +02:00
|
|
|
// m_shutdown_* are already handled, skip.
|
|
|
|
return;
|
2017-04-15 23:19:18 +02:00
|
|
|
} else if (delay > 0.0f) {
|
2017-04-21 19:31:59 +02:00
|
|
|
// Positive delay, tell the clients when the server will shut down
|
2017-04-15 23:19:18 +02:00
|
|
|
std::wstringstream ws;
|
|
|
|
|
|
|
|
ws << L"*** Server shutting down in "
|
2018-06-11 13:43:12 +02:00
|
|
|
<< duration_to_string(myround(delay)).c_str()
|
2017-04-15 23:19:18 +02:00
|
|
|
<< ".";
|
|
|
|
|
|
|
|
infostream << wide_to_utf8(ws.str()).c_str() << std::endl;
|
|
|
|
SendChatMessage(PEER_ID_INEXISTENT, ws.str());
|
|
|
|
}
|
2018-06-11 13:43:12 +02:00
|
|
|
|
2018-06-13 21:58:34 +02:00
|
|
|
m_shutdown_state.trigger(delay, msg, reconnect);
|
2017-04-15 23:19:18 +02:00
|
|
|
}
|
|
|
|
|
2017-09-27 19:47:36 +02:00
|
|
|
PlayerSAO* Server::emergePlayer(const char *name, session_t peer_id, u16 proto_version)
|
2011-01-15 02:28:19 +01:00
|
|
|
{
|
|
|
|
/*
|
|
|
|
Try to get an existing player
|
|
|
|
*/
|
2016-10-08 17:56:38 +02:00
|
|
|
RemotePlayer *player = m_env->getPlayer(name);
|
2012-01-24 00:00:26 +01:00
|
|
|
|
2012-03-19 03:04:16 +01:00
|
|
|
// If player is already connected, cancel
|
2017-09-30 12:00:05 +02:00
|
|
|
if (player && player->getPeerId() != PEER_ID_INEXISTENT) {
|
2012-03-19 03:04:16 +01:00
|
|
|
infostream<<"emergePlayer(): Player already connected"<<std::endl;
|
|
|
|
return NULL;
|
2011-01-15 02:28:19 +01:00
|
|
|
}
|
2011-01-17 23:26:09 +01:00
|
|
|
|
|
|
|
/*
|
|
|
|
If player with the wanted peer_id already exists, cancel.
|
|
|
|
*/
|
2017-08-15 09:30:31 +02:00
|
|
|
if (m_env->getPlayer(peer_id)) {
|
2011-10-16 13:57:53 +02:00
|
|
|
infostream<<"emergePlayer(): Player with wrong name but same"
|
2011-01-17 23:26:09 +01:00
|
|
|
" peer_id already exists"<<std::endl;
|
|
|
|
return NULL;
|
|
|
|
}
|
2012-03-19 03:04:16 +01:00
|
|
|
|
2014-05-30 22:04:07 +02:00
|
|
|
if (!player) {
|
2017-04-23 14:35:08 +02:00
|
|
|
player = new RemotePlayer(name, idef());
|
2012-03-19 03:04:16 +01:00
|
|
|
}
|
2011-01-15 02:28:19 +01:00
|
|
|
|
2017-04-23 14:35:08 +02:00
|
|
|
bool newplayer = false;
|
2015-07-17 16:40:41 +02:00
|
|
|
|
2017-04-23 14:35:08 +02:00
|
|
|
// Load player
|
|
|
|
PlayerSAO *playersao = m_env->loadPlayer(player, &newplayer, peer_id, isSingleplayer());
|
2013-05-10 03:10:33 +02:00
|
|
|
|
2017-04-23 14:35:08 +02:00
|
|
|
// Complete init with server parts
|
|
|
|
playersao->finalize(player, getPlayerEffectivePrivs(player->getName()));
|
|
|
|
player->protocol_version = proto_version;
|
2011-01-15 02:28:19 +01:00
|
|
|
|
2012-03-19 03:04:16 +01:00
|
|
|
/* Run scripts */
|
2014-05-30 22:04:07 +02:00
|
|
|
if (newplayer) {
|
2013-05-25 00:51:02 +02:00
|
|
|
m_script->on_newplayer(playersao);
|
2014-05-30 22:04:07 +02:00
|
|
|
}
|
2012-03-19 03:04:16 +01:00
|
|
|
|
|
|
|
return playersao;
|
2011-01-15 02:28:19 +01:00
|
|
|
}
|
|
|
|
|
2017-02-08 00:15:55 +01:00
|
|
|
bool Server::registerModStorage(ModMetadata *storage)
|
|
|
|
{
|
|
|
|
if (m_mod_storages.find(storage->getModName()) != m_mod_storages.end()) {
|
|
|
|
errorstream << "Unable to register same mod storage twice. Storage name: "
|
|
|
|
<< storage->getModName() << std::endl;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
m_mod_storages[storage->getModName()] = storage;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Server::unregisterModStorage(const std::string &name)
|
|
|
|
{
|
2017-06-04 21:00:04 +02:00
|
|
|
std::unordered_map<std::string, ModMetadata *>::const_iterator it = m_mod_storages.find(name);
|
2017-02-08 00:15:55 +01:00
|
|
|
if (it != m_mod_storages.end()) {
|
|
|
|
// Save unconditionaly on unregistration
|
|
|
|
it->second->save(getModStoragePath());
|
|
|
|
m_mod_storages.erase(name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-02-15 15:11:24 +01:00
|
|
|
void dedicated_server_loop(Server &server, bool &kill)
|
2011-01-23 16:29:15 +01:00
|
|
|
{
|
2012-03-11 03:15:45 +01:00
|
|
|
verbosestream<<"dedicated_server_loop()"<<std::endl;
|
2011-05-30 23:15:43 +02:00
|
|
|
|
|
|
|
IntervalLimiter m_profiler_interval;
|
2011-01-23 16:29:15 +01:00
|
|
|
|
2017-06-11 13:58:43 +02:00
|
|
|
static thread_local const float steplen =
|
|
|
|
g_settings->getFloat("dedicated_server_step");
|
|
|
|
static thread_local const float profiler_print_interval =
|
2015-09-02 21:21:55 +02:00
|
|
|
g_settings->getFloat("profiler_print_interval");
|
|
|
|
|
|
|
|
for(;;) {
|
2011-01-23 16:29:15 +01:00
|
|
|
// This is kind of a hack but can be done like this
|
|
|
|
// because server.step() is very light
|
2011-05-30 23:15:43 +02:00
|
|
|
{
|
2011-10-12 12:53:38 +02:00
|
|
|
ScopeProfiler sp(g_profiler, "dedicated server sleep");
|
2012-03-07 20:44:53 +01:00
|
|
|
sleep_ms((int)(steplen*1000.0));
|
2011-05-30 23:15:43 +02:00
|
|
|
}
|
2012-03-07 20:44:53 +01:00
|
|
|
server.step(steplen);
|
2011-01-23 16:29:15 +01:00
|
|
|
|
2018-06-13 21:58:34 +02:00
|
|
|
if (server.isShutdownRequested() || kill)
|
2017-03-19 18:17:59 +01:00
|
|
|
break;
|
|
|
|
|
2011-05-30 23:15:43 +02:00
|
|
|
/*
|
|
|
|
Profiler
|
|
|
|
*/
|
2015-09-02 21:21:55 +02:00
|
|
|
if (profiler_print_interval != 0) {
|
2012-03-07 20:44:53 +01:00
|
|
|
if(m_profiler_interval.step(steplen, profiler_print_interval))
|
2011-05-30 23:15:43 +02:00
|
|
|
{
|
2011-10-16 13:57:53 +02:00
|
|
|
infostream<<"Profiler:"<<std::endl;
|
|
|
|
g_profiler->print(infostream);
|
2011-10-12 12:53:38 +02:00
|
|
|
g_profiler->clear();
|
2011-05-30 23:15:43 +02:00
|
|
|
}
|
|
|
|
}
|
2011-01-23 16:29:15 +01:00
|
|
|
}
|
2015-10-15 19:04:15 +02:00
|
|
|
|
|
|
|
infostream << "Dedicated server quitting" << std::endl;
|
|
|
|
#if USE_CURL
|
|
|
|
if (g_settings->getBool("server_announce"))
|
|
|
|
ServerList::sendAnnounce(ServerList::AA_DELETE,
|
|
|
|
server.m_bind_addr.getPort());
|
|
|
|
#endif
|
2011-01-23 16:29:15 +01:00
|
|
|
}
|
2017-09-26 00:11:20 +02:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Mod channels
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
bool Server::joinModChannel(const std::string &channel)
|
|
|
|
{
|
|
|
|
return m_modchannel_mgr->joinChannel(channel, PEER_ID_SERVER) &&
|
|
|
|
m_modchannel_mgr->setChannelState(channel, MODCHANNEL_STATE_READ_WRITE);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Server::leaveModChannel(const std::string &channel)
|
|
|
|
{
|
|
|
|
return m_modchannel_mgr->leaveChannel(channel, PEER_ID_SERVER);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Server::sendModChannelMessage(const std::string &channel, const std::string &message)
|
|
|
|
{
|
|
|
|
if (!m_modchannel_mgr->canWriteOnChannel(channel))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
broadcastModChannelMessage(channel, message, PEER_ID_SERVER);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
ModChannel* Server::getModChannel(const std::string &channel)
|
|
|
|
{
|
|
|
|
return m_modchannel_mgr->getModChannel(channel);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Server::broadcastModChannelMessage(const std::string &channel,
|
2017-09-27 23:48:06 +02:00
|
|
|
const std::string &message, session_t from_peer)
|
2017-09-26 00:11:20 +02:00
|
|
|
{
|
|
|
|
const std::vector<u16> &peers = m_modchannel_mgr->getChannelPeers(channel);
|
|
|
|
if (peers.empty())
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (message.size() > STRING_MAX_LEN) {
|
|
|
|
warningstream << "ModChannel message too long, dropping before sending "
|
|
|
|
<< " (" << message.size() << " > " << STRING_MAX_LEN << ", channel: "
|
|
|
|
<< channel << ")" << std::endl;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string sender;
|
|
|
|
if (from_peer != PEER_ID_SERVER) {
|
|
|
|
sender = getPlayerName(from_peer);
|
|
|
|
}
|
|
|
|
|
|
|
|
NetworkPacket resp_pkt(TOCLIENT_MODCHANNEL_MSG,
|
|
|
|
2 + channel.size() + 2 + sender.size() + 2 + message.size());
|
|
|
|
resp_pkt << channel << sender << message;
|
2017-09-27 19:47:36 +02:00
|
|
|
for (session_t peer_id : peers) {
|
2017-09-26 00:11:20 +02:00
|
|
|
// Ignore sender
|
|
|
|
if (peer_id == from_peer)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
Send(peer_id, &resp_pkt);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (from_peer != PEER_ID_SERVER) {
|
|
|
|
m_script->on_modchannel_message(channel, sender, message);
|
|
|
|
}
|
|
|
|
}
|