2013-08-11 04:09:45 +02:00
|
|
|
/*
|
|
|
|
Minetest
|
|
|
|
Copyright (C) 2013 celeron55, Perttu Ahola <celeron55@gmail.com>
|
|
|
|
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
|
|
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
|
|
|
|
(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
|
|
|
|
GNU Lesser General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU Lesser General Public License along
|
|
|
|
with this program; if not, write to the Free Software Foundation, Inc.,
|
|
|
|
51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "lua_api/l_server.h"
|
|
|
|
#include "lua_api/l_internal.h"
|
|
|
|
#include "common/c_converter.h"
|
|
|
|
#include "common/c_content.h"
|
2014-09-06 02:08:51 +02:00
|
|
|
#include "cpp_api/s_base.h"
|
2020-06-13 19:03:26 +02:00
|
|
|
#include "cpp_api/s_security.h"
|
2021-08-28 12:15:12 +02:00
|
|
|
#include "scripting_server.h"
|
2013-08-11 04:09:45 +02:00
|
|
|
#include "server.h"
|
|
|
|
#include "environment.h"
|
2017-08-18 19:25:07 +02:00
|
|
|
#include "remoteplayer.h"
|
2014-08-23 13:24:37 +02:00
|
|
|
#include "log.h"
|
2015-07-06 18:53:30 +02:00
|
|
|
#include <algorithm>
|
2013-08-11 04:09:45 +02:00
|
|
|
|
|
|
|
// request_shutdown()
|
|
|
|
int ModApiServer::l_request_shutdown(lua_State *L)
|
|
|
|
{
|
2015-10-25 05:45:18 +01:00
|
|
|
NO_MAP_LOCK_REQUIRED;
|
2015-07-17 16:40:41 +02:00
|
|
|
const char *msg = lua_tolstring(L, 1, NULL);
|
2018-06-30 17:11:38 +02:00
|
|
|
bool reconnect = readParam<bool>(L, 2);
|
2017-04-15 23:19:18 +02:00
|
|
|
float seconds_before_shutdown = lua_tonumber(L, 3);
|
|
|
|
getServer(L)->requestShutdown(msg ? msg : "", reconnect, seconds_before_shutdown);
|
2013-08-11 04:09:45 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// get_server_status()
|
|
|
|
int ModApiServer::l_get_server_status(lua_State *L)
|
|
|
|
{
|
|
|
|
NO_MAP_LOCK_REQUIRED;
|
2021-01-29 14:03:27 +01:00
|
|
|
lua_pushstring(L, getServer(L)->getStatusString().c_str());
|
2013-08-11 04:09:45 +02:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2016-11-02 17:36:58 +01:00
|
|
|
// get_server_uptime()
|
|
|
|
int ModApiServer::l_get_server_uptime(lua_State *L)
|
|
|
|
{
|
|
|
|
NO_MAP_LOCK_REQUIRED;
|
|
|
|
lua_pushnumber(L, getServer(L)->getUptime());
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2021-10-05 14:35:55 +02:00
|
|
|
// get_server_max_lag()
|
|
|
|
int ModApiServer::l_get_server_max_lag(lua_State *L)
|
|
|
|
{
|
|
|
|
NO_MAP_LOCK_REQUIRED;
|
2022-04-09 14:47:59 +02:00
|
|
|
GET_ENV_PTR;
|
|
|
|
lua_pushnumber(L, env->getMaxLagEstimate());
|
2021-10-05 14:35:55 +02:00
|
|
|
return 1;
|
|
|
|
}
|
2016-11-02 17:36:58 +01:00
|
|
|
|
2015-07-25 07:43:32 +02:00
|
|
|
// print(text)
|
|
|
|
int ModApiServer::l_print(lua_State *L)
|
|
|
|
{
|
|
|
|
NO_MAP_LOCK_REQUIRED;
|
|
|
|
std::string text;
|
|
|
|
text = luaL_checkstring(L, 1);
|
|
|
|
getServer(L)->printToConsoleOnly(text);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2013-08-11 04:09:45 +02:00
|
|
|
// chat_send_all(text)
|
|
|
|
int ModApiServer::l_chat_send_all(lua_State *L)
|
|
|
|
{
|
|
|
|
NO_MAP_LOCK_REQUIRED;
|
|
|
|
const char *text = luaL_checkstring(L, 1);
|
|
|
|
// Get server from registry
|
|
|
|
Server *server = getServer(L);
|
|
|
|
// Send
|
2018-06-09 16:31:35 +02:00
|
|
|
server->notifyPlayers(utf8_to_wide(text));
|
2013-08-11 04:09:45 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-02-25 02:41:40 +01:00
|
|
|
// chat_send_player(name, text)
|
2013-08-11 04:09:45 +02:00
|
|
|
int ModApiServer::l_chat_send_player(lua_State *L)
|
|
|
|
{
|
|
|
|
NO_MAP_LOCK_REQUIRED;
|
|
|
|
const char *name = luaL_checkstring(L, 1);
|
|
|
|
const char *text = luaL_checkstring(L, 2);
|
|
|
|
|
|
|
|
// Get server from registry
|
|
|
|
Server *server = getServer(L);
|
|
|
|
// Send
|
2018-06-09 16:31:35 +02:00
|
|
|
server->notifyPlayer(name, utf8_to_wide(text));
|
2013-08-11 04:09:45 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// get_player_privs(name, text)
|
|
|
|
int ModApiServer::l_get_player_privs(lua_State *L)
|
|
|
|
{
|
|
|
|
NO_MAP_LOCK_REQUIRED;
|
|
|
|
const char *name = luaL_checkstring(L, 1);
|
|
|
|
// Get server from registry
|
|
|
|
Server *server = getServer(L);
|
|
|
|
// Do it
|
|
|
|
lua_newtable(L);
|
|
|
|
int table = lua_gettop(L);
|
|
|
|
std::set<std::string> privs_s = server->getPlayerEffectivePrivs(name);
|
2017-08-17 08:11:39 +02:00
|
|
|
for (const std::string &privs_ : privs_s) {
|
2013-08-11 04:09:45 +02:00
|
|
|
lua_pushboolean(L, true);
|
2017-08-17 08:11:39 +02:00
|
|
|
lua_setfield(L, table, privs_.c_str());
|
2013-08-11 04:09:45 +02:00
|
|
|
}
|
|
|
|
lua_pushvalue(L, table);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// get_player_ip()
|
|
|
|
int ModApiServer::l_get_player_ip(lua_State *L)
|
|
|
|
{
|
|
|
|
NO_MAP_LOCK_REQUIRED;
|
2021-01-31 20:36:47 +01:00
|
|
|
|
|
|
|
Server *server = getServer(L);
|
|
|
|
|
|
|
|
const char *name = luaL_checkstring(L, 1);
|
|
|
|
RemotePlayer *player = server->getEnv().getPlayer(name);
|
|
|
|
if (!player) {
|
2013-08-11 04:09:45 +02:00
|
|
|
lua_pushnil(L); // no such player
|
|
|
|
return 1;
|
|
|
|
}
|
2021-01-31 20:36:47 +01:00
|
|
|
|
|
|
|
lua_pushstring(L, server->getPeerAddress(player->getPeerId()).serializeString().c_str());
|
|
|
|
return 1;
|
2013-08-11 04:09:45 +02:00
|
|
|
}
|
|
|
|
|
2017-04-27 11:49:44 +02:00
|
|
|
// get_player_information(name)
|
2014-02-13 20:17:42 +01:00
|
|
|
int ModApiServer::l_get_player_information(lua_State *L)
|
|
|
|
{
|
|
|
|
NO_MAP_LOCK_REQUIRED;
|
2020-05-01 21:44:28 +02:00
|
|
|
|
|
|
|
Server *server = getServer(L);
|
|
|
|
|
|
|
|
const char *name = luaL_checkstring(L, 1);
|
|
|
|
RemotePlayer *player = server->getEnv().getPlayer(name);
|
|
|
|
if (!player) {
|
2014-02-13 20:17:42 +01:00
|
|
|
lua_pushnil(L); // no such player
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2021-01-31 20:36:47 +01:00
|
|
|
/*
|
|
|
|
Be careful not to introduce a depdendency on the connection to
|
|
|
|
the peer here. This function is >>REQUIRED<< to still be able to return
|
|
|
|
values even when the peer unexpectedly disappears.
|
|
|
|
Hence all the ConInfo values here are optional.
|
|
|
|
*/
|
2020-05-01 21:44:28 +02:00
|
|
|
|
|
|
|
auto getConInfo = [&] (con::rtt_stat_type type, float *value) -> bool {
|
|
|
|
return server->getClientConInfo(player->getPeerId(), type, value);
|
|
|
|
};
|
|
|
|
|
2021-01-31 20:36:47 +01:00
|
|
|
float min_rtt, max_rtt, avg_rtt, min_jitter, max_jitter, avg_jitter;
|
2020-05-01 21:44:28 +02:00
|
|
|
bool have_con_info =
|
|
|
|
getConInfo(con::MIN_RTT, &min_rtt) &&
|
|
|
|
getConInfo(con::MAX_RTT, &max_rtt) &&
|
|
|
|
getConInfo(con::AVG_RTT, &avg_rtt) &&
|
|
|
|
getConInfo(con::MIN_JITTER, &min_jitter) &&
|
|
|
|
getConInfo(con::MAX_JITTER, &max_jitter) &&
|
|
|
|
getConInfo(con::AVG_JITTER, &avg_jitter);
|
|
|
|
|
2021-01-31 20:36:47 +01:00
|
|
|
ClientInfo info;
|
|
|
|
if (!server->getClientInfo(player->getPeerId(), info)) {
|
|
|
|
warningstream << FUNCTION_NAME << ": no client info?!" << std::endl;
|
2020-05-01 21:44:28 +02:00
|
|
|
lua_pushnil(L); // error
|
|
|
|
return 1;
|
2014-02-13 20:17:42 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
lua_newtable(L);
|
|
|
|
int table = lua_gettop(L);
|
|
|
|
|
|
|
|
lua_pushstring(L,"address");
|
2021-01-31 20:36:47 +01:00
|
|
|
lua_pushstring(L, info.addr.serializeString().c_str());
|
2014-02-13 20:17:42 +01:00
|
|
|
lua_settable(L, table);
|
|
|
|
|
|
|
|
lua_pushstring(L,"ip_version");
|
2021-01-31 20:36:47 +01:00
|
|
|
if (info.addr.getFamily() == AF_INET) {
|
2014-02-13 20:17:42 +01:00
|
|
|
lua_pushnumber(L, 4);
|
2021-01-31 20:36:47 +01:00
|
|
|
} else if (info.addr.getFamily() == AF_INET6) {
|
2014-02-13 20:17:42 +01:00
|
|
|
lua_pushnumber(L, 6);
|
|
|
|
} else {
|
|
|
|
lua_pushnumber(L, 0);
|
|
|
|
}
|
|
|
|
lua_settable(L, table);
|
|
|
|
|
2020-05-01 21:44:28 +02:00
|
|
|
if (have_con_info) { // may be missing
|
|
|
|
lua_pushstring(L, "min_rtt");
|
|
|
|
lua_pushnumber(L, min_rtt);
|
|
|
|
lua_settable(L, table);
|
2014-02-13 20:17:42 +01:00
|
|
|
|
2020-05-01 21:44:28 +02:00
|
|
|
lua_pushstring(L, "max_rtt");
|
|
|
|
lua_pushnumber(L, max_rtt);
|
|
|
|
lua_settable(L, table);
|
2014-02-13 20:17:42 +01:00
|
|
|
|
2020-05-01 21:44:28 +02:00
|
|
|
lua_pushstring(L, "avg_rtt");
|
|
|
|
lua_pushnumber(L, avg_rtt);
|
|
|
|
lua_settable(L, table);
|
2014-02-13 20:17:42 +01:00
|
|
|
|
2020-05-01 21:44:28 +02:00
|
|
|
lua_pushstring(L, "min_jitter");
|
|
|
|
lua_pushnumber(L, min_jitter);
|
|
|
|
lua_settable(L, table);
|
2014-02-13 20:17:42 +01:00
|
|
|
|
2020-05-01 21:44:28 +02:00
|
|
|
lua_pushstring(L, "max_jitter");
|
|
|
|
lua_pushnumber(L, max_jitter);
|
|
|
|
lua_settable(L, table);
|
2014-02-13 20:17:42 +01:00
|
|
|
|
2020-05-01 21:44:28 +02:00
|
|
|
lua_pushstring(L, "avg_jitter");
|
|
|
|
lua_pushnumber(L, avg_jitter);
|
|
|
|
lua_settable(L, table);
|
|
|
|
}
|
2014-02-13 20:17:42 +01:00
|
|
|
|
|
|
|
lua_pushstring(L,"connection_uptime");
|
2021-01-31 20:36:47 +01:00
|
|
|
lua_pushnumber(L, info.uptime);
|
2014-02-13 20:17:42 +01:00
|
|
|
lua_settable(L, table);
|
|
|
|
|
2017-04-27 11:49:44 +02:00
|
|
|
lua_pushstring(L,"protocol_version");
|
2021-01-31 20:36:47 +01:00
|
|
|
lua_pushnumber(L, info.prot_vers);
|
2017-04-27 11:49:44 +02:00
|
|
|
lua_settable(L, table);
|
2017-08-17 08:11:39 +02:00
|
|
|
|
2019-09-10 21:18:42 +02:00
|
|
|
lua_pushstring(L, "formspec_version");
|
|
|
|
lua_pushnumber(L, player->formspec_version);
|
|
|
|
lua_settable(L, table);
|
|
|
|
|
2020-04-25 07:20:00 +02:00
|
|
|
lua_pushstring(L, "lang_code");
|
2021-01-31 20:36:47 +01:00
|
|
|
lua_pushstring(L, info.lang_code.c_str());
|
2020-04-25 07:20:00 +02:00
|
|
|
lua_settable(L, table);
|
|
|
|
|
2014-02-13 20:17:42 +01:00
|
|
|
#ifndef NDEBUG
|
|
|
|
lua_pushstring(L,"serialization_version");
|
2021-01-31 20:36:47 +01:00
|
|
|
lua_pushnumber(L, info.ser_vers);
|
2014-02-13 20:17:42 +01:00
|
|
|
lua_settable(L, table);
|
|
|
|
|
|
|
|
lua_pushstring(L,"major");
|
2021-01-31 20:36:47 +01:00
|
|
|
lua_pushnumber(L, info.major);
|
2014-02-13 20:17:42 +01:00
|
|
|
lua_settable(L, table);
|
|
|
|
|
|
|
|
lua_pushstring(L,"minor");
|
2021-01-31 20:36:47 +01:00
|
|
|
lua_pushnumber(L, info.minor);
|
2014-02-13 20:17:42 +01:00
|
|
|
lua_settable(L, table);
|
|
|
|
|
|
|
|
lua_pushstring(L,"patch");
|
2021-01-31 20:36:47 +01:00
|
|
|
lua_pushnumber(L, info.patch);
|
2014-02-13 20:17:42 +01:00
|
|
|
lua_settable(L, table);
|
|
|
|
|
|
|
|
lua_pushstring(L,"version_string");
|
2021-01-31 20:36:47 +01:00
|
|
|
lua_pushstring(L, info.vers_string.c_str());
|
2014-02-13 20:17:42 +01:00
|
|
|
lua_settable(L, table);
|
|
|
|
|
|
|
|
lua_pushstring(L,"state");
|
2021-01-31 20:36:47 +01:00
|
|
|
lua_pushstring(L, ClientInterface::state2Name(info.state).c_str());
|
2014-02-13 20:17:42 +01:00
|
|
|
lua_settable(L, table);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2013-08-11 04:09:45 +02:00
|
|
|
// get_ban_list()
|
|
|
|
int ModApiServer::l_get_ban_list(lua_State *L)
|
|
|
|
{
|
|
|
|
NO_MAP_LOCK_REQUIRED;
|
|
|
|
lua_pushstring(L, getServer(L)->getBanDescription("").c_str());
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// get_ban_description()
|
|
|
|
int ModApiServer::l_get_ban_description(lua_State *L)
|
|
|
|
{
|
|
|
|
NO_MAP_LOCK_REQUIRED;
|
|
|
|
const char * ip_or_name = luaL_checkstring(L, 1);
|
|
|
|
lua_pushstring(L, getServer(L)->getBanDescription(std::string(ip_or_name)).c_str());
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// ban_player()
|
|
|
|
int ModApiServer::l_ban_player(lua_State *L)
|
|
|
|
{
|
|
|
|
NO_MAP_LOCK_REQUIRED;
|
2021-01-31 20:36:47 +01:00
|
|
|
|
2021-10-05 21:53:47 +02:00
|
|
|
if (!getEnv(L))
|
|
|
|
throw LuaError("Can't ban player before server has started up");
|
2021-01-31 20:36:47 +01:00
|
|
|
|
2021-10-05 21:53:47 +02:00
|
|
|
Server *server = getServer(L);
|
2021-01-31 20:36:47 +01:00
|
|
|
const char *name = luaL_checkstring(L, 1);
|
|
|
|
RemotePlayer *player = server->getEnv().getPlayer(name);
|
|
|
|
if (!player) {
|
2013-08-11 04:09:45 +02:00
|
|
|
lua_pushboolean(L, false); // no such player
|
|
|
|
return 1;
|
|
|
|
}
|
2021-01-31 20:36:47 +01:00
|
|
|
|
|
|
|
std::string ip_str = server->getPeerAddress(player->getPeerId()).serializeString();
|
|
|
|
server->setIpBanned(ip_str, name);
|
2013-08-11 04:09:45 +02:00
|
|
|
lua_pushboolean(L, true);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2021-11-26 20:19:40 +01:00
|
|
|
// disconnect_player(name, [reason]) -> success
|
|
|
|
int ModApiServer::l_disconnect_player(lua_State *L)
|
2014-01-26 18:40:25 +01:00
|
|
|
{
|
|
|
|
NO_MAP_LOCK_REQUIRED;
|
2021-10-05 21:53:47 +02:00
|
|
|
|
|
|
|
if (!getEnv(L))
|
|
|
|
throw LuaError("Can't kick player before server has started up");
|
|
|
|
|
2014-01-26 18:40:25 +01:00
|
|
|
const char *name = luaL_checkstring(L, 1);
|
2021-11-26 20:19:40 +01:00
|
|
|
std::string message;
|
2014-01-26 18:40:25 +01:00
|
|
|
if (lua_isstring(L, 2))
|
2021-11-26 20:19:40 +01:00
|
|
|
message.append(readParam<std::string>(L, 2));
|
2014-01-26 18:40:25 +01:00
|
|
|
else
|
2021-11-26 20:19:40 +01:00
|
|
|
message.append("Disconnected.");
|
2016-10-08 12:21:41 +02:00
|
|
|
|
2022-04-27 19:55:13 +02:00
|
|
|
Server *server = getServer(L);
|
|
|
|
|
|
|
|
RemotePlayer *player = server->getEnv().getPlayer(name);
|
|
|
|
if (!player) {
|
2014-01-26 18:40:25 +01:00
|
|
|
lua_pushboolean(L, false); // No such player
|
|
|
|
return 1;
|
|
|
|
}
|
2022-04-27 19:55:13 +02:00
|
|
|
|
|
|
|
server->DenyAccess(player->getPeerId(), SERVER_ACCESSDENIED_CUSTOM_STRING, message);
|
2014-01-26 18:40:25 +01:00
|
|
|
lua_pushboolean(L, true);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2017-04-23 14:35:08 +02:00
|
|
|
int ModApiServer::l_remove_player(lua_State *L)
|
|
|
|
{
|
|
|
|
NO_MAP_LOCK_REQUIRED;
|
|
|
|
std::string name = luaL_checkstring(L, 1);
|
|
|
|
ServerEnvironment *s_env = dynamic_cast<ServerEnvironment *>(getEnv(L));
|
2021-10-05 21:53:47 +02:00
|
|
|
if (!s_env)
|
|
|
|
throw LuaError("Can't remove player before server has started up");
|
2017-04-23 14:35:08 +02:00
|
|
|
|
|
|
|
RemotePlayer *player = s_env->getPlayer(name.c_str());
|
|
|
|
if (!player)
|
|
|
|
lua_pushinteger(L, s_env->removePlayerFromDatabase(name) ? 0 : 1);
|
|
|
|
else
|
|
|
|
lua_pushinteger(L, 2);
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2013-08-11 04:09:45 +02:00
|
|
|
// unban_player_or_ip()
|
|
|
|
int ModApiServer::l_unban_player_or_ip(lua_State *L)
|
|
|
|
{
|
|
|
|
NO_MAP_LOCK_REQUIRED;
|
|
|
|
const char * ip_or_name = luaL_checkstring(L, 1);
|
|
|
|
getServer(L)->unsetIpBanned(ip_or_name);
|
|
|
|
lua_pushboolean(L, true);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// show_formspec(playername,formname,formspec)
|
|
|
|
int ModApiServer::l_show_formspec(lua_State *L)
|
|
|
|
{
|
|
|
|
NO_MAP_LOCK_REQUIRED;
|
|
|
|
const char *playername = luaL_checkstring(L, 1);
|
|
|
|
const char *formname = luaL_checkstring(L, 2);
|
|
|
|
const char *formspec = luaL_checkstring(L, 3);
|
|
|
|
|
|
|
|
if(getServer(L)->showFormspec(playername,formspec,formname))
|
|
|
|
{
|
|
|
|
lua_pushboolean(L, true);
|
|
|
|
}else{
|
|
|
|
lua_pushboolean(L, false);
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// get_current_modname()
|
|
|
|
int ModApiServer::l_get_current_modname(lua_State *L)
|
|
|
|
{
|
|
|
|
NO_MAP_LOCK_REQUIRED;
|
2015-08-25 07:00:56 +02:00
|
|
|
lua_rawgeti(L, LUA_REGISTRYINDEX, CUSTOM_RIDX_CURRENT_MOD_NAME);
|
2013-08-11 04:09:45 +02:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// get_modpath(modname)
|
|
|
|
int ModApiServer::l_get_modpath(lua_State *L)
|
|
|
|
{
|
|
|
|
NO_MAP_LOCK_REQUIRED;
|
|
|
|
std::string modname = luaL_checkstring(L, 1);
|
2022-04-09 14:47:59 +02:00
|
|
|
const ModSpec *mod = getGameDef(L)->getModSpec(modname);
|
|
|
|
if (!mod)
|
2013-08-11 04:09:45 +02:00
|
|
|
lua_pushnil(L);
|
2022-04-09 14:47:59 +02:00
|
|
|
else
|
|
|
|
lua_pushstring(L, mod->path.c_str());
|
2013-08-11 04:09:45 +02:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// get_modnames()
|
|
|
|
// the returned list is sorted alphabetically for you
|
|
|
|
int ModApiServer::l_get_modnames(lua_State *L)
|
|
|
|
{
|
|
|
|
NO_MAP_LOCK_REQUIRED;
|
2013-11-05 18:06:15 +01:00
|
|
|
|
2013-08-11 04:09:45 +02:00
|
|
|
// Get a list of mods
|
2015-03-05 20:12:54 +01:00
|
|
|
std::vector<std::string> modlist;
|
2022-04-09 14:47:59 +02:00
|
|
|
for (auto &it : getGameDef(L)->getMods())
|
|
|
|
modlist.emplace_back(it.name);
|
2013-08-11 04:09:45 +02:00
|
|
|
|
2015-03-05 20:12:54 +01:00
|
|
|
std::sort(modlist.begin(), modlist.end());
|
2013-08-11 04:09:45 +02:00
|
|
|
|
|
|
|
// Package them up for Lua
|
2015-03-05 20:12:54 +01:00
|
|
|
lua_createtable(L, modlist.size(), 0);
|
2022-04-09 14:47:59 +02:00
|
|
|
auto iter = modlist.begin();
|
2017-04-25 10:17:53 +02:00
|
|
|
for (u16 i = 0; iter != modlist.end(); ++iter) {
|
2013-11-30 18:11:07 +01:00
|
|
|
lua_pushstring(L, iter->c_str());
|
|
|
|
lua_rawseti(L, -2, ++i);
|
2013-08-11 04:09:45 +02:00
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// get_worldpath()
|
|
|
|
int ModApiServer::l_get_worldpath(lua_State *L)
|
|
|
|
{
|
|
|
|
NO_MAP_LOCK_REQUIRED;
|
2022-04-09 14:47:59 +02:00
|
|
|
const Server *srv = getServer(L);
|
|
|
|
lua_pushstring(L, srv->getWorldPath().c_str());
|
2013-08-11 04:09:45 +02:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2020-01-25 21:19:29 +01:00
|
|
|
// sound_play(spec, parameters, [ephemeral])
|
2013-08-11 04:09:45 +02:00
|
|
|
int ModApiServer::l_sound_play(lua_State *L)
|
|
|
|
{
|
|
|
|
NO_MAP_LOCK_REQUIRED;
|
|
|
|
SimpleSoundSpec spec;
|
|
|
|
read_soundspec(L, 1, spec);
|
|
|
|
ServerSoundParams params;
|
|
|
|
read_server_sound_params(L, 2, params);
|
2020-01-25 21:19:29 +01:00
|
|
|
bool ephemeral = lua_gettop(L) > 2 && readParam<bool>(L, 3);
|
|
|
|
if (ephemeral) {
|
|
|
|
getServer(L)->playSound(spec, params, true);
|
|
|
|
lua_pushnil(L);
|
|
|
|
} else {
|
|
|
|
s32 handle = getServer(L)->playSound(spec, params);
|
|
|
|
lua_pushinteger(L, handle);
|
|
|
|
}
|
2013-08-11 04:09:45 +02:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// sound_stop(handle)
|
|
|
|
int ModApiServer::l_sound_stop(lua_State *L)
|
|
|
|
{
|
|
|
|
NO_MAP_LOCK_REQUIRED;
|
2020-01-25 21:19:29 +01:00
|
|
|
s32 handle = luaL_checkinteger(L, 1);
|
2013-08-11 04:09:45 +02:00
|
|
|
getServer(L)->stopSound(handle);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2016-07-10 07:08:26 +02:00
|
|
|
int ModApiServer::l_sound_fade(lua_State *L)
|
|
|
|
{
|
|
|
|
NO_MAP_LOCK_REQUIRED;
|
|
|
|
s32 handle = luaL_checkinteger(L, 1);
|
2018-06-04 22:38:07 +02:00
|
|
|
float step = readParam<float>(L, 2);
|
|
|
|
float gain = readParam<float>(L, 3);
|
2016-07-10 07:08:26 +02:00
|
|
|
getServer(L)->fadeSound(handle, step, gain);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2020-06-13 19:03:26 +02:00
|
|
|
// dynamic_add_media(filepath)
|
2021-09-09 16:51:35 +02:00
|
|
|
int ModApiServer::l_dynamic_add_media(lua_State *L)
|
2020-06-13 19:03:26 +02:00
|
|
|
{
|
|
|
|
NO_MAP_LOCK_REQUIRED;
|
|
|
|
|
|
|
|
if (!getEnv(L))
|
|
|
|
throw LuaError("Dynamic media cannot be added before server has started up");
|
2021-09-09 16:51:35 +02:00
|
|
|
Server *server = getServer(L);
|
2020-06-13 19:03:26 +02:00
|
|
|
|
2021-09-09 16:51:35 +02:00
|
|
|
std::string filepath;
|
|
|
|
std::string to_player;
|
|
|
|
bool ephemeral = false;
|
2020-06-13 19:03:26 +02:00
|
|
|
|
2021-09-09 16:51:35 +02:00
|
|
|
if (lua_istable(L, 1)) {
|
|
|
|
getstringfield(L, 1, "filepath", filepath);
|
|
|
|
getstringfield(L, 1, "to_player", to_player);
|
|
|
|
getboolfield(L, 1, "ephemeral", ephemeral);
|
2021-01-30 14:35:34 +01:00
|
|
|
} else {
|
2021-09-09 16:51:35 +02:00
|
|
|
filepath = readParam<std::string>(L, 1);
|
2021-01-30 14:35:34 +01:00
|
|
|
}
|
2021-09-09 16:51:35 +02:00
|
|
|
if (filepath.empty())
|
|
|
|
luaL_typerror(L, 1, "non-empty string");
|
|
|
|
luaL_checktype(L, 2, LUA_TFUNCTION);
|
|
|
|
|
|
|
|
CHECK_SECURE_PATH(L, filepath.c_str(), false);
|
|
|
|
|
2021-12-18 20:36:43 +01:00
|
|
|
u32 token = server->getScriptIface()->allocateDynamicMediaCallback(L, 2);
|
2021-09-09 16:51:35 +02:00
|
|
|
|
|
|
|
bool ok = server->dynamicAddMedia(filepath, token, to_player, ephemeral);
|
|
|
|
if (!ok)
|
|
|
|
server->getScriptIface()->freeDynamicMediaCallback(token);
|
|
|
|
lua_pushboolean(L, ok);
|
2021-01-30 14:35:34 +01:00
|
|
|
|
2020-06-13 19:03:26 +02:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2013-08-11 04:09:45 +02:00
|
|
|
// is_singleplayer()
|
|
|
|
int ModApiServer::l_is_singleplayer(lua_State *L)
|
|
|
|
{
|
|
|
|
NO_MAP_LOCK_REQUIRED;
|
2022-04-09 14:47:59 +02:00
|
|
|
const Server *srv = getServer(L);
|
|
|
|
lua_pushboolean(L, srv->isSingleplayer());
|
2013-08-11 04:09:45 +02:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// notify_authentication_modified(name)
|
|
|
|
int ModApiServer::l_notify_authentication_modified(lua_State *L)
|
|
|
|
{
|
|
|
|
NO_MAP_LOCK_REQUIRED;
|
2017-08-17 08:11:39 +02:00
|
|
|
std::string name;
|
2013-08-11 04:09:45 +02:00
|
|
|
if(lua_isstring(L, 1))
|
2018-06-30 17:11:38 +02:00
|
|
|
name = readParam<std::string>(L, 1);
|
2013-08-11 04:09:45 +02:00
|
|
|
getServer(L)->reportPrivsModified(name);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ModApiServer::Initialize(lua_State *L, int top)
|
|
|
|
{
|
|
|
|
API_FCT(request_shutdown);
|
|
|
|
API_FCT(get_server_status);
|
2016-11-02 17:36:58 +01:00
|
|
|
API_FCT(get_server_uptime);
|
2021-10-05 14:35:55 +02:00
|
|
|
API_FCT(get_server_max_lag);
|
2013-08-11 04:09:45 +02:00
|
|
|
API_FCT(get_worldpath);
|
|
|
|
API_FCT(is_singleplayer);
|
|
|
|
|
|
|
|
API_FCT(get_current_modname);
|
|
|
|
API_FCT(get_modpath);
|
|
|
|
API_FCT(get_modnames);
|
|
|
|
|
2015-07-25 07:43:32 +02:00
|
|
|
API_FCT(print);
|
|
|
|
|
2013-08-11 04:09:45 +02:00
|
|
|
API_FCT(chat_send_all);
|
|
|
|
API_FCT(chat_send_player);
|
|
|
|
API_FCT(show_formspec);
|
|
|
|
API_FCT(sound_play);
|
|
|
|
API_FCT(sound_stop);
|
2016-07-10 07:08:26 +02:00
|
|
|
API_FCT(sound_fade);
|
2021-09-09 16:51:35 +02:00
|
|
|
API_FCT(dynamic_add_media);
|
2013-08-11 04:09:45 +02:00
|
|
|
|
2014-02-13 20:17:42 +01:00
|
|
|
API_FCT(get_player_information);
|
2013-08-11 04:09:45 +02:00
|
|
|
API_FCT(get_player_privs);
|
|
|
|
API_FCT(get_player_ip);
|
|
|
|
API_FCT(get_ban_list);
|
|
|
|
API_FCT(get_ban_description);
|
|
|
|
API_FCT(ban_player);
|
2021-11-26 20:19:40 +01:00
|
|
|
API_FCT(disconnect_player);
|
2017-04-23 14:35:08 +02:00
|
|
|
API_FCT(remove_player);
|
2013-08-11 04:09:45 +02:00
|
|
|
API_FCT(unban_player_or_ip);
|
|
|
|
API_FCT(notify_authentication_modified);
|
|
|
|
}
|