2011-11-11 12:33:17 -05:00
|
|
|
/*
|
2013-02-24 12:40:43 -05:00
|
|
|
Minetest
|
2013-02-24 13:38:45 -05:00
|
|
|
Copyright (C) 2013 celeron55, Perttu Ahola <celeron55@gmail.com>
|
2011-11-11 12:33:17 -05:00
|
|
|
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
2012-06-05 10:56:56 -04: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
|
2011-11-11 12:33:17 -05: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 10:56:56 -04:00
|
|
|
GNU Lesser General Public License for more details.
|
2011-11-11 12:33:17 -05:00
|
|
|
|
2012-06-05 10:56:56 -04:00
|
|
|
You should have received a copy of the GNU Lesser General Public License along
|
2011-11-11 12:33:17 -05:00
|
|
|
with this program; if not, write to the Free Software Foundation, Inc.,
|
|
|
|
51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "scriptapi.h"
|
|
|
|
|
|
|
|
#include <iostream>
|
2011-11-28 10:11:14 -05:00
|
|
|
#include <list>
|
2011-11-11 12:33:17 -05:00
|
|
|
extern "C" {
|
|
|
|
#include <lua.h>
|
|
|
|
#include <lualib.h>
|
|
|
|
#include <lauxlib.h>
|
|
|
|
}
|
|
|
|
|
2011-11-25 20:37:09 -05:00
|
|
|
#include "settings.h" // For accessing g_settings
|
2013-02-23 13:06:57 -05:00
|
|
|
#include "main.h" // For g_settings
|
|
|
|
#include "biome.h"
|
2013-03-24 01:43:38 -04:00
|
|
|
#include "emerge.h"
|
2013-02-23 13:06:57 -05:00
|
|
|
#include "script.h"
|
2012-07-26 15:06:45 -04:00
|
|
|
#include "rollback.h"
|
2011-11-11 12:33:17 -05:00
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
#include "scriptapi_types.h"
|
|
|
|
#include "scriptapi_env.h"
|
|
|
|
#include "scriptapi_nodetimer.h"
|
|
|
|
#include "scriptapi_inventory.h"
|
|
|
|
#include "scriptapi_nodemeta.h"
|
|
|
|
#include "scriptapi_object.h"
|
|
|
|
#include "scriptapi_noise.h"
|
|
|
|
#include "scriptapi_common.h"
|
|
|
|
#include "scriptapi_item.h"
|
|
|
|
#include "scriptapi_content.h"
|
|
|
|
#include "scriptapi_craft.h"
|
2013-01-23 12:32:02 -05:00
|
|
|
#include "scriptapi_particles.h"
|
2013-02-23 13:06:57 -05:00
|
|
|
|
|
|
|
/*****************************************************************************/
|
|
|
|
/* Mod related */
|
|
|
|
/*****************************************************************************/
|
2011-11-11 19:25:30 -05:00
|
|
|
|
2011-12-02 19:45:55 -05:00
|
|
|
class ModNameStorer
|
|
|
|
{
|
|
|
|
private:
|
|
|
|
lua_State *L;
|
|
|
|
public:
|
|
|
|
ModNameStorer(lua_State *L_, const std::string modname):
|
|
|
|
L(L_)
|
|
|
|
{
|
|
|
|
// Store current modname in registry
|
|
|
|
lua_pushstring(L, modname.c_str());
|
|
|
|
lua_setfield(L, LUA_REGISTRYINDEX, "minetest_current_modname");
|
|
|
|
}
|
|
|
|
~ModNameStorer()
|
|
|
|
{
|
|
|
|
// Clear current modname in registry
|
|
|
|
lua_pushnil(L);
|
|
|
|
lua_setfield(L, LUA_REGISTRYINDEX, "minetest_current_modname");
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
bool scriptapi_loadmod(lua_State *L, const std::string &scriptpath,
|
|
|
|
const std::string &modname)
|
2011-11-17 04:22:24 -05:00
|
|
|
{
|
2013-02-23 13:06:57 -05:00
|
|
|
ModNameStorer modnamestorer(L, modname);
|
2011-11-17 04:22:24 -05:00
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
if(!string_allowed(modname, "abcdefghijklmnopqrstuvwxyz"
|
|
|
|
"0123456789_")){
|
|
|
|
errorstream<<"Error loading mod \""<<modname
|
|
|
|
<<"\": modname does not follow naming conventions: "
|
|
|
|
<<"Only chararacters [a-z0-9_] are allowed."<<std::endl;
|
|
|
|
return false;
|
|
|
|
}
|
2011-11-25 07:58:42 -05:00
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
bool success = false;
|
2011-11-25 09:34:12 -05:00
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
try{
|
|
|
|
success = script_load(L, scriptpath.c_str());
|
2011-11-25 07:58:42 -05:00
|
|
|
}
|
2013-02-23 13:06:57 -05:00
|
|
|
catch(LuaError &e){
|
|
|
|
errorstream<<"Error loading mod \""<<modname
|
|
|
|
<<"\": "<<e.what()<<std::endl;
|
2011-11-25 07:58:42 -05:00
|
|
|
}
|
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
return success;
|
2011-11-25 07:58:42 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
/*****************************************************************************/
|
|
|
|
/* Auth */
|
|
|
|
/*****************************************************************************/
|
2011-12-06 08:55:05 -05:00
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
/*
|
|
|
|
Privileges
|
|
|
|
*/
|
|
|
|
static void read_privileges(lua_State *L, int index,
|
|
|
|
std::set<std::string> &result)
|
2011-11-25 07:58:42 -05:00
|
|
|
{
|
2013-02-23 13:06:57 -05:00
|
|
|
result.clear();
|
|
|
|
lua_pushnil(L);
|
|
|
|
if(index < 0)
|
|
|
|
index -= 1;
|
|
|
|
while(lua_next(L, index) != 0){
|
|
|
|
// key at index -2 and value at index -1
|
|
|
|
std::string key = luaL_checkstring(L, -2);
|
|
|
|
bool value = lua_toboolean(L, -1);
|
|
|
|
if(value)
|
|
|
|
result.insert(key);
|
|
|
|
// removes value, keeps key for next iteration
|
|
|
|
lua_pop(L, 1);
|
|
|
|
}
|
2011-11-25 07:58:42 -05:00
|
|
|
}
|
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
static void get_auth_handler(lua_State *L)
|
2011-11-25 07:58:42 -05:00
|
|
|
{
|
2013-02-23 13:06:57 -05:00
|
|
|
lua_getglobal(L, "minetest");
|
|
|
|
lua_getfield(L, -1, "registered_auth_handler");
|
|
|
|
if(lua_isnil(L, -1)){
|
|
|
|
lua_pop(L, 1);
|
|
|
|
lua_getfield(L, -1, "builtin_auth_handler");
|
|
|
|
}
|
|
|
|
if(lua_type(L, -1) != LUA_TTABLE)
|
|
|
|
throw LuaError(L, "Authentication handler table not valid");
|
2011-11-25 07:58:42 -05:00
|
|
|
}
|
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
bool scriptapi_get_auth(lua_State *L, const std::string &playername,
|
|
|
|
std::string *dst_password, std::set<std::string> *dst_privs)
|
2011-11-26 21:31:05 -05:00
|
|
|
{
|
2013-02-23 13:06:57 -05:00
|
|
|
realitycheck(L);
|
|
|
|
assert(lua_checkstack(L, 20));
|
|
|
|
StackUnroller stack_unroller(L);
|
2011-11-26 21:31:05 -05:00
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
get_auth_handler(L);
|
|
|
|
lua_getfield(L, -1, "get_auth");
|
|
|
|
if(lua_type(L, -1) != LUA_TFUNCTION)
|
|
|
|
throw LuaError(L, "Authentication handler missing get_auth");
|
|
|
|
lua_pushstring(L, playername.c_str());
|
|
|
|
if(lua_pcall(L, 1, 1, 0))
|
|
|
|
script_error(L, "error: %s", lua_tostring(L, -1));
|
2011-11-25 07:58:42 -05:00
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
// nil = login not allowed
|
|
|
|
if(lua_isnil(L, -1))
|
|
|
|
return false;
|
|
|
|
luaL_checktype(L, -1, LUA_TTABLE);
|
2011-11-25 07:58:42 -05:00
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
std::string password;
|
|
|
|
bool found = getstringfield(L, -1, "password", password);
|
|
|
|
if(!found)
|
|
|
|
throw LuaError(L, "Authentication handler didn't return password");
|
|
|
|
if(dst_password)
|
|
|
|
*dst_password = password;
|
2011-11-25 07:58:42 -05:00
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
lua_getfield(L, -1, "privileges");
|
|
|
|
if(!lua_istable(L, -1))
|
|
|
|
throw LuaError(L,
|
|
|
|
"Authentication handler didn't return privilege table");
|
|
|
|
if(dst_privs)
|
|
|
|
read_privileges(L, -1, *dst_privs);
|
|
|
|
lua_pop(L, 1);
|
2011-11-25 07:58:42 -05:00
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
return true;
|
2011-11-25 07:58:42 -05:00
|
|
|
}
|
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
void scriptapi_create_auth(lua_State *L, const std::string &playername,
|
|
|
|
const std::string &password)
|
2012-01-20 18:11:44 -05:00
|
|
|
{
|
2013-02-23 13:06:57 -05:00
|
|
|
realitycheck(L);
|
|
|
|
assert(lua_checkstack(L, 20));
|
|
|
|
StackUnroller stack_unroller(L);
|
2012-01-20 18:11:44 -05:00
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
get_auth_handler(L);
|
|
|
|
lua_getfield(L, -1, "create_auth");
|
|
|
|
if(lua_type(L, -1) != LUA_TFUNCTION)
|
|
|
|
throw LuaError(L, "Authentication handler missing create_auth");
|
|
|
|
lua_pushstring(L, playername.c_str());
|
|
|
|
lua_pushstring(L, password.c_str());
|
|
|
|
if(lua_pcall(L, 2, 0, 0))
|
|
|
|
script_error(L, "error: %s", lua_tostring(L, -1));
|
2011-12-01 16:33:48 -05:00
|
|
|
}
|
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
bool scriptapi_set_password(lua_State *L, const std::string &playername,
|
|
|
|
const std::string &password)
|
2012-01-20 18:11:44 -05:00
|
|
|
{
|
2013-02-23 13:06:57 -05:00
|
|
|
realitycheck(L);
|
|
|
|
assert(lua_checkstack(L, 20));
|
|
|
|
StackUnroller stack_unroller(L);
|
2012-01-20 18:11:44 -05:00
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
get_auth_handler(L);
|
|
|
|
lua_getfield(L, -1, "set_password");
|
|
|
|
if(lua_type(L, -1) != LUA_TFUNCTION)
|
|
|
|
throw LuaError(L, "Authentication handler missing set_password");
|
|
|
|
lua_pushstring(L, playername.c_str());
|
|
|
|
lua_pushstring(L, password.c_str());
|
|
|
|
if(lua_pcall(L, 2, 1, 0))
|
|
|
|
script_error(L, "error: %s", lua_tostring(L, -1));
|
|
|
|
return lua_toboolean(L, -1);
|
2011-12-03 20:45:02 -05:00
|
|
|
}
|
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
/*****************************************************************************/
|
|
|
|
/* Misc */
|
|
|
|
/*****************************************************************************/
|
2011-11-30 16:38:18 -05:00
|
|
|
/*
|
2013-02-23 13:06:57 -05:00
|
|
|
Groups
|
2011-11-30 16:38:18 -05:00
|
|
|
*/
|
2013-02-23 13:06:57 -05:00
|
|
|
void read_groups(lua_State *L, int index,
|
|
|
|
std::map<std::string, int> &result)
|
2012-06-15 20:40:45 -04:00
|
|
|
{
|
2013-02-23 13:06:57 -05:00
|
|
|
if (!lua_istable(L,index))
|
|
|
|
return;
|
|
|
|
result.clear();
|
|
|
|
lua_pushnil(L);
|
|
|
|
if(index < 0)
|
|
|
|
index -= 1;
|
|
|
|
while(lua_next(L, index) != 0){
|
|
|
|
// key at index -2 and value at index -1
|
|
|
|
std::string name = luaL_checkstring(L, -2);
|
|
|
|
int rating = luaL_checkinteger(L, -1);
|
|
|
|
result[name] = rating;
|
|
|
|
// removes value, keeps key for next iteration
|
|
|
|
lua_pop(L, 1);
|
|
|
|
}
|
|
|
|
}
|
2012-06-15 20:40:45 -04:00
|
|
|
|
2012-12-18 13:23:16 -05:00
|
|
|
struct EnumString es_BiomeTerrainType[] =
|
|
|
|
{
|
|
|
|
{BIOME_TERRAIN_NORMAL, "normal"},
|
|
|
|
{BIOME_TERRAIN_LIQUID, "liquid"},
|
|
|
|
{BIOME_TERRAIN_NETHER, "nether"},
|
|
|
|
{BIOME_TERRAIN_AETHER, "aether"},
|
|
|
|
{BIOME_TERRAIN_FLAT, "flat"},
|
|
|
|
{0, NULL},
|
|
|
|
};
|
|
|
|
|
2013-03-24 01:43:38 -04:00
|
|
|
struct EnumString es_OreType[] =
|
|
|
|
{
|
|
|
|
{ORE_SCATTER, "scatter"},
|
|
|
|
{ORE_SHEET, "sheet"},
|
|
|
|
{ORE_CLAYLIKE, "claylike"},
|
|
|
|
{0, NULL},
|
|
|
|
};
|
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
/*****************************************************************************/
|
|
|
|
/* Parameters */
|
|
|
|
/*****************************************************************************/
|
2011-11-12 04:59:56 -05:00
|
|
|
/*
|
2013-02-23 13:06:57 -05:00
|
|
|
DigParams
|
2011-11-12 04:59:56 -05:00
|
|
|
*/
|
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
static void set_dig_params(lua_State *L, int table,
|
|
|
|
const DigParams ¶ms)
|
2011-11-25 19:26:19 -05:00
|
|
|
{
|
2013-02-23 13:06:57 -05:00
|
|
|
setboolfield(L, table, "diggable", params.diggable);
|
|
|
|
setfloatfield(L, table, "time", params.time);
|
|
|
|
setintfield(L, table, "wear", params.wear);
|
2011-11-25 19:26:19 -05:00
|
|
|
}
|
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
static void push_dig_params(lua_State *L,
|
|
|
|
const DigParams ¶ms)
|
2011-11-11 12:33:17 -05:00
|
|
|
{
|
2013-02-23 13:06:57 -05:00
|
|
|
lua_newtable(L);
|
|
|
|
set_dig_params(L, -1, params);
|
2011-12-06 08:55:05 -05:00
|
|
|
}
|
2011-11-11 17:46:05 -05:00
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
/*
|
|
|
|
HitParams
|
|
|
|
*/
|
|
|
|
|
|
|
|
static void set_hit_params(lua_State *L, int table,
|
|
|
|
const HitParams ¶ms)
|
2012-01-12 00:10:39 -05:00
|
|
|
{
|
2013-02-23 13:06:57 -05:00
|
|
|
setintfield(L, table, "hp", params.hp);
|
|
|
|
setintfield(L, table, "wear", params.wear);
|
2012-01-12 00:10:39 -05:00
|
|
|
}
|
2011-11-11 12:33:17 -05:00
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
static void push_hit_params(lua_State *L,
|
|
|
|
const HitParams ¶ms)
|
2012-01-12 00:10:39 -05:00
|
|
|
{
|
|
|
|
lua_newtable(L);
|
2013-02-23 13:06:57 -05:00
|
|
|
set_hit_params(L, -1, params);
|
2012-01-12 00:10:39 -05:00
|
|
|
}
|
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
/*
|
|
|
|
ServerSoundParams
|
|
|
|
*/
|
2012-01-12 00:10:39 -05:00
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
static void read_server_sound_params(lua_State *L, int index,
|
|
|
|
ServerSoundParams ¶ms)
|
2012-03-18 23:25:09 -04:00
|
|
|
{
|
2013-02-23 13:06:57 -05:00
|
|
|
if(index < 0)
|
|
|
|
index = lua_gettop(L) + 1 + index;
|
|
|
|
// Clear
|
|
|
|
params = ServerSoundParams();
|
2012-03-18 23:25:09 -04:00
|
|
|
if(lua_istable(L, index)){
|
2013-02-23 13:06:57 -05:00
|
|
|
getfloatfield(L, index, "gain", params.gain);
|
|
|
|
getstringfield(L, index, "to_player", params.to_player);
|
|
|
|
lua_getfield(L, index, "pos");
|
|
|
|
if(!lua_isnil(L, -1)){
|
|
|
|
v3f p = read_v3f(L, -1)*BS;
|
|
|
|
params.pos = p;
|
|
|
|
params.type = ServerSoundParams::SSP_POSITIONAL;
|
2012-03-18 23:25:09 -04:00
|
|
|
}
|
2013-02-23 13:06:57 -05:00
|
|
|
lua_pop(L, 1);
|
|
|
|
lua_getfield(L, index, "object");
|
|
|
|
if(!lua_isnil(L, -1)){
|
|
|
|
ObjectRef *ref = ObjectRef::checkobject(L, -1);
|
|
|
|
ServerActiveObject *sao = ObjectRef::getobject(ref);
|
|
|
|
if(sao){
|
|
|
|
params.object = sao->getId();
|
|
|
|
params.type = ServerSoundParams::SSP_OBJECT;
|
2012-03-18 23:25:09 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
lua_pop(L, 1);
|
2013-02-23 13:06:57 -05:00
|
|
|
params.max_hear_distance = BS*getfloatfield_default(L, index,
|
|
|
|
"max_hear_distance", params.max_hear_distance/BS);
|
|
|
|
getboolfield(L, index, "loop", params.loop);
|
2012-03-18 23:25:09 -04:00
|
|
|
}
|
2012-01-20 18:11:44 -05:00
|
|
|
}
|
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
/*****************************************************************************/
|
|
|
|
/* callbacks */
|
|
|
|
/*****************************************************************************/
|
2013-01-27 23:30:37 -05:00
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
// Push the list of callbacks (a lua table).
|
|
|
|
// Then push nargs arguments.
|
|
|
|
// Then call this function, which
|
|
|
|
// - runs the callbacks
|
|
|
|
// - removes the table and arguments from the lua stack
|
|
|
|
// - pushes the return value, computed depending on mode
|
|
|
|
void scriptapi_run_callbacks(lua_State *L, int nargs,
|
|
|
|
RunCallbacksMode mode)
|
2012-02-28 12:45:23 -05:00
|
|
|
{
|
2013-02-23 13:06:57 -05:00
|
|
|
// Insert the return value into the lua stack, below the table
|
|
|
|
assert(lua_gettop(L) >= nargs + 1);
|
2012-02-28 12:45:23 -05:00
|
|
|
lua_pushnil(L);
|
2013-02-23 13:06:57 -05:00
|
|
|
lua_insert(L, -(nargs + 1) - 1);
|
|
|
|
// Stack now looks like this:
|
|
|
|
// ... <return value = nil> <table> <arg#1> <arg#2> ... <arg#n>
|
2012-02-28 12:45:23 -05:00
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
int rv = lua_gettop(L) - nargs - 1;
|
|
|
|
int table = rv + 1;
|
|
|
|
int arg = table + 1;
|
|
|
|
|
|
|
|
luaL_checktype(L, table, LUA_TTABLE);
|
|
|
|
|
|
|
|
// Foreach
|
2012-03-30 11:42:18 -04:00
|
|
|
lua_pushnil(L);
|
2013-02-23 13:06:57 -05:00
|
|
|
bool first_loop = true;
|
|
|
|
while(lua_next(L, table) != 0){
|
2012-03-30 11:42:18 -04:00
|
|
|
// key at index -2 and value at index -1
|
2013-02-23 13:06:57 -05:00
|
|
|
luaL_checktype(L, -1, LUA_TFUNCTION);
|
|
|
|
// Call function
|
|
|
|
for(int i = 0; i < nargs; i++)
|
|
|
|
lua_pushvalue(L, arg+i);
|
|
|
|
if(lua_pcall(L, nargs, 1, 0))
|
|
|
|
script_error(L, "error: %s", lua_tostring(L, -1));
|
2012-01-12 00:10:39 -05:00
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
// Move return value to designated space in stack
|
|
|
|
// Or pop it
|
|
|
|
if(first_loop){
|
|
|
|
// Result of first callback is always moved
|
|
|
|
lua_replace(L, rv);
|
|
|
|
first_loop = false;
|
|
|
|
} else {
|
|
|
|
// Otherwise, what happens depends on the mode
|
|
|
|
if(mode == RUN_CALLBACKS_MODE_FIRST)
|
2012-02-28 12:45:23 -05:00
|
|
|
lua_pop(L, 1);
|
2013-02-23 13:06:57 -05:00
|
|
|
else if(mode == RUN_CALLBACKS_MODE_LAST)
|
|
|
|
lua_replace(L, rv);
|
|
|
|
else if(mode == RUN_CALLBACKS_MODE_AND ||
|
|
|
|
mode == RUN_CALLBACKS_MODE_AND_SC){
|
|
|
|
if((bool)lua_toboolean(L, rv) == true &&
|
|
|
|
(bool)lua_toboolean(L, -1) == false)
|
|
|
|
lua_replace(L, rv);
|
|
|
|
else
|
|
|
|
lua_pop(L, 1);
|
2012-02-28 12:45:23 -05:00
|
|
|
}
|
2013-02-23 13:06:57 -05:00
|
|
|
else if(mode == RUN_CALLBACKS_MODE_OR ||
|
|
|
|
mode == RUN_CALLBACKS_MODE_OR_SC){
|
|
|
|
if((bool)lua_toboolean(L, rv) == false &&
|
|
|
|
(bool)lua_toboolean(L, -1) == true)
|
|
|
|
lua_replace(L, rv);
|
|
|
|
else
|
|
|
|
lua_pop(L, 1);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
assert(0);
|
2012-02-28 12:45:23 -05:00
|
|
|
}
|
2013-02-23 13:06:57 -05:00
|
|
|
|
|
|
|
// Handle short circuit modes
|
|
|
|
if(mode == RUN_CALLBACKS_MODE_AND_SC &&
|
|
|
|
(bool)lua_toboolean(L, rv) == false)
|
|
|
|
break;
|
|
|
|
else if(mode == RUN_CALLBACKS_MODE_OR_SC &&
|
|
|
|
(bool)lua_toboolean(L, rv) == true)
|
|
|
|
break;
|
|
|
|
|
|
|
|
// value removed, keep key for next iteration
|
2012-02-28 12:45:23 -05:00
|
|
|
}
|
2012-01-12 00:10:39 -05:00
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
// Remove stuff from stack, leaving only the return value
|
|
|
|
lua_settop(L, rv);
|
|
|
|
|
|
|
|
// Fix return value in case no callbacks were called
|
|
|
|
if(first_loop){
|
|
|
|
if(mode == RUN_CALLBACKS_MODE_AND ||
|
|
|
|
mode == RUN_CALLBACKS_MODE_AND_SC){
|
|
|
|
lua_pop(L, 1);
|
|
|
|
lua_pushboolean(L, true);
|
|
|
|
}
|
|
|
|
else if(mode == RUN_CALLBACKS_MODE_OR ||
|
|
|
|
mode == RUN_CALLBACKS_MODE_OR_SC){
|
|
|
|
lua_pop(L, 1);
|
|
|
|
lua_pushboolean(L, false);
|
2012-02-28 12:45:23 -05:00
|
|
|
}
|
|
|
|
}
|
2012-01-12 00:10:39 -05:00
|
|
|
}
|
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
bool scriptapi_on_chat_message(lua_State *L, const std::string &name,
|
|
|
|
const std::string &message)
|
2012-01-20 18:11:44 -05:00
|
|
|
{
|
2013-02-23 13:06:57 -05:00
|
|
|
realitycheck(L);
|
|
|
|
assert(lua_checkstack(L, 20));
|
|
|
|
StackUnroller stack_unroller(L);
|
2012-07-22 10:10:58 -04:00
|
|
|
|
|
|
|
// Get minetest.registered_on_chat_messages
|
|
|
|
lua_getglobal(L, "minetest");
|
2013-02-23 13:06:57 -05:00
|
|
|
lua_getfield(L, -1, "registered_on_chat_messages");
|
2012-07-22 10:10:58 -04:00
|
|
|
// Call callbacks
|
2013-02-23 13:06:57 -05:00
|
|
|
lua_pushstring(L, name.c_str());
|
|
|
|
lua_pushstring(L, message.c_str());
|
|
|
|
scriptapi_run_callbacks(L, 2, RUN_CALLBACKS_MODE_OR_SC);
|
|
|
|
bool ate = lua_toboolean(L, -1);
|
|
|
|
return ate;
|
2011-11-29 10:15:18 -05:00
|
|
|
}
|
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
void scriptapi_on_shutdown(lua_State *L)
|
2011-11-12 10:46:06 -05:00
|
|
|
{
|
|
|
|
realitycheck(L);
|
|
|
|
assert(lua_checkstack(L, 20));
|
|
|
|
StackUnroller stack_unroller(L);
|
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
// Get registered shutdown hooks
|
|
|
|
lua_getglobal(L, "minetest");
|
|
|
|
lua_getfield(L, -1, "registered_on_shutdown");
|
|
|
|
// Call callbacks
|
|
|
|
scriptapi_run_callbacks(L, 0, RUN_CALLBACKS_MODE_FIRST);
|
2011-11-17 04:22:24 -05:00
|
|
|
}
|
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
void scriptapi_on_newplayer(lua_State *L, ServerActiveObject *player)
|
2011-11-17 04:22:24 -05:00
|
|
|
{
|
|
|
|
realitycheck(L);
|
|
|
|
assert(lua_checkstack(L, 20));
|
|
|
|
StackUnroller stack_unroller(L);
|
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
// Get minetest.registered_on_newplayers
|
|
|
|
lua_getglobal(L, "minetest");
|
|
|
|
lua_getfield(L, -1, "registered_on_newplayers");
|
|
|
|
// Call callbacks
|
|
|
|
objectref_get_or_create(L, player);
|
|
|
|
scriptapi_run_callbacks(L, 1, RUN_CALLBACKS_MODE_FIRST);
|
2011-11-17 04:22:24 -05:00
|
|
|
}
|
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
void scriptapi_on_dieplayer(lua_State *L, ServerActiveObject *player)
|
2012-06-01 10:24:54 -04:00
|
|
|
{
|
|
|
|
realitycheck(L);
|
|
|
|
assert(lua_checkstack(L, 20));
|
|
|
|
StackUnroller stack_unroller(L);
|
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
// Get minetest.registered_on_dieplayers
|
|
|
|
lua_getglobal(L, "minetest");
|
|
|
|
lua_getfield(L, -1, "registered_on_dieplayers");
|
|
|
|
// Call callbacks
|
|
|
|
objectref_get_or_create(L, player);
|
|
|
|
scriptapi_run_callbacks(L, 1, RUN_CALLBACKS_MODE_FIRST);
|
2012-06-01 10:24:54 -04:00
|
|
|
}
|
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
bool scriptapi_on_respawnplayer(lua_State *L, ServerActiveObject *player)
|
2012-06-01 10:24:54 -04:00
|
|
|
{
|
|
|
|
realitycheck(L);
|
|
|
|
assert(lua_checkstack(L, 20));
|
|
|
|
StackUnroller stack_unroller(L);
|
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
// Get minetest.registered_on_respawnplayers
|
|
|
|
lua_getglobal(L, "minetest");
|
|
|
|
lua_getfield(L, -1, "registered_on_respawnplayers");
|
|
|
|
// Call callbacks
|
|
|
|
objectref_get_or_create(L, player);
|
|
|
|
scriptapi_run_callbacks(L, 1, RUN_CALLBACKS_MODE_OR);
|
|
|
|
bool positioning_handled_by_some = lua_toboolean(L, -1);
|
|
|
|
return positioning_handled_by_some;
|
2012-06-01 10:24:54 -04:00
|
|
|
}
|
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
void scriptapi_on_joinplayer(lua_State *L, ServerActiveObject *player)
|
2012-06-05 16:51:37 -04:00
|
|
|
{
|
|
|
|
realitycheck(L);
|
|
|
|
assert(lua_checkstack(L, 20));
|
|
|
|
StackUnroller stack_unroller(L);
|
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
// Get minetest.registered_on_joinplayers
|
|
|
|
lua_getglobal(L, "minetest");
|
|
|
|
lua_getfield(L, -1, "registered_on_joinplayers");
|
|
|
|
// Call callbacks
|
|
|
|
objectref_get_or_create(L, player);
|
|
|
|
scriptapi_run_callbacks(L, 1, RUN_CALLBACKS_MODE_FIRST);
|
2012-06-05 16:51:37 -04:00
|
|
|
}
|
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
void scriptapi_on_leaveplayer(lua_State *L, ServerActiveObject *player)
|
2012-07-17 09:00:04 -04:00
|
|
|
{
|
|
|
|
realitycheck(L);
|
|
|
|
assert(lua_checkstack(L, 20));
|
|
|
|
StackUnroller stack_unroller(L);
|
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
// Get minetest.registered_on_leaveplayers
|
|
|
|
lua_getglobal(L, "minetest");
|
|
|
|
lua_getfield(L, -1, "registered_on_leaveplayers");
|
|
|
|
// Call callbacks
|
|
|
|
objectref_get_or_create(L, player);
|
|
|
|
scriptapi_run_callbacks(L, 1, RUN_CALLBACKS_MODE_FIRST);
|
2012-07-17 09:00:04 -04:00
|
|
|
}
|
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
/*
|
|
|
|
player
|
|
|
|
*/
|
|
|
|
void scriptapi_on_player_receive_fields(lua_State *L,
|
|
|
|
ServerActiveObject *player,
|
2012-06-01 13:51:15 -04:00
|
|
|
const std::string &formname,
|
2013-02-23 13:06:57 -05:00
|
|
|
const std::map<std::string, std::string> &fields)
|
2012-06-01 13:51:15 -04:00
|
|
|
{
|
|
|
|
realitycheck(L);
|
|
|
|
assert(lua_checkstack(L, 20));
|
|
|
|
StackUnroller stack_unroller(L);
|
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
// Get minetest.registered_on_chat_messages
|
|
|
|
lua_getglobal(L, "minetest");
|
|
|
|
lua_getfield(L, -1, "registered_on_player_receive_fields");
|
|
|
|
// Call callbacks
|
2012-06-01 13:51:15 -04:00
|
|
|
// param 1
|
2013-02-23 13:06:57 -05:00
|
|
|
objectref_get_or_create(L, player);
|
2012-06-01 13:51:15 -04:00
|
|
|
// param 2
|
|
|
|
lua_pushstring(L, formname.c_str());
|
|
|
|
// param 3
|
|
|
|
lua_newtable(L);
|
|
|
|
for(std::map<std::string, std::string>::const_iterator
|
|
|
|
i = fields.begin(); i != fields.end(); i++){
|
|
|
|
const std::string &name = i->first;
|
|
|
|
const std::string &value = i->second;
|
|
|
|
lua_pushstring(L, name.c_str());
|
|
|
|
lua_pushlstring(L, value.c_str(), value.size());
|
|
|
|
lua_settable(L, -3);
|
|
|
|
}
|
2013-02-23 13:06:57 -05:00
|
|
|
scriptapi_run_callbacks(L, 3, RUN_CALLBACKS_MODE_OR_SC);
|
2012-06-01 13:51:15 -04:00
|
|
|
}
|
2013-02-23 13:06:57 -05:00
|
|
|
/*****************************************************************************/
|
|
|
|
/* Api functions */
|
|
|
|
/*****************************************************************************/
|
2012-06-01 13:51:15 -04:00
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
// debug(text)
|
|
|
|
// Writes a line to dstream
|
|
|
|
static int l_debug(lua_State *L)
|
|
|
|
{
|
|
|
|
std::string text = lua_tostring(L, 1);
|
|
|
|
dstream << text << std::endl;
|
|
|
|
return 0;
|
|
|
|
}
|
2012-07-24 19:36:54 -04:00
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
// log([level,] text)
|
|
|
|
// Writes a line to the logger.
|
|
|
|
// The one-argument version logs to infostream.
|
|
|
|
// The two-argument version accept a log level: error, action, info, or verbose.
|
|
|
|
static int l_log(lua_State *L)
|
2012-07-24 19:36:54 -04:00
|
|
|
{
|
2013-02-23 13:06:57 -05:00
|
|
|
std::string text;
|
|
|
|
LogMessageLevel level = LMT_INFO;
|
|
|
|
if(lua_isnone(L, 2))
|
|
|
|
{
|
|
|
|
text = lua_tostring(L, 1);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
std::string levelname = luaL_checkstring(L, 1);
|
|
|
|
text = luaL_checkstring(L, 2);
|
|
|
|
if(levelname == "error")
|
|
|
|
level = LMT_ERROR;
|
|
|
|
else if(levelname == "action")
|
|
|
|
level = LMT_ACTION;
|
|
|
|
else if(levelname == "verbose")
|
|
|
|
level = LMT_VERBOSE;
|
|
|
|
}
|
|
|
|
log_printline(level, text);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// request_shutdown()
|
|
|
|
static int l_request_shutdown(lua_State *L)
|
|
|
|
{
|
|
|
|
get_server(L)->requestShutdown();
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// get_server_status()
|
|
|
|
static int l_get_server_status(lua_State *L)
|
|
|
|
{
|
|
|
|
lua_pushstring(L, wide_to_narrow(get_server(L)->getStatusString()).c_str());
|
|
|
|
return 1;
|
|
|
|
}
|
2012-07-24 19:36:54 -04:00
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
// register_biome_groups({frequencies})
|
|
|
|
static int l_register_biome_groups(lua_State *L)
|
|
|
|
{
|
|
|
|
luaL_checktype(L, 1, LUA_TTABLE);
|
|
|
|
int index = 1;
|
2012-07-24 19:36:54 -04:00
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
BiomeDefManager *bmgr = get_server(L)->getBiomeDef();
|
|
|
|
if (!bmgr) {
|
|
|
|
verbosestream << "register_biome_groups: BiomeDefManager not active" << std::endl;
|
2012-07-24 19:36:54 -04:00
|
|
|
return 0;
|
2013-02-23 13:06:57 -05:00
|
|
|
}
|
2012-07-24 19:36:54 -04:00
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
lua_pushnil(L);
|
|
|
|
for (int i = 1; lua_next(L, index) != 0; i++) {
|
|
|
|
bmgr->addBiomeGroup(lua_tonumber(L, -1));
|
|
|
|
lua_pop(L, 1);
|
|
|
|
}
|
|
|
|
lua_pop(L, 1);
|
2012-07-24 19:36:54 -04:00
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
return 0;
|
2012-07-24 19:36:54 -04:00
|
|
|
}
|
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
// register_biome({lots of stuff})
|
|
|
|
static int l_register_biome(lua_State *L)
|
2012-07-24 19:36:54 -04:00
|
|
|
{
|
2013-02-23 13:06:57 -05:00
|
|
|
luaL_checktype(L, 1, LUA_TTABLE);
|
|
|
|
int index = 1, groupid;
|
|
|
|
std::string nodename;
|
2012-07-24 19:36:54 -04:00
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
IWritableNodeDefManager *ndef = get_server(L)->getWritableNodeDefManager();
|
|
|
|
BiomeDefManager *bmgr = get_server(L)->getBiomeDef();
|
|
|
|
if (!bmgr) {
|
|
|
|
verbosestream << "register_biome: BiomeDefManager not active" << std::endl;
|
2012-07-24 19:36:54 -04:00
|
|
|
return 0;
|
2013-02-23 13:06:57 -05:00
|
|
|
}
|
2012-07-24 19:36:54 -04:00
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
groupid = getintfield_default(L, index, "group_id", 0);
|
2012-07-24 19:36:54 -04:00
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
enum BiomeTerrainType terrain = (BiomeTerrainType)getenumfield(L, index,
|
|
|
|
"terrain_type", es_BiomeTerrainType, BIOME_TERRAIN_NORMAL);
|
|
|
|
Biome *b = bmgr->createBiome(terrain);
|
|
|
|
|
|
|
|
b->name = getstringfield_default(L, index, "name", "");
|
|
|
|
|
|
|
|
if (getstringfield(L, index, "node_top", nodename))
|
|
|
|
b->n_top = MapNode(ndef->getId(nodename));
|
|
|
|
else
|
|
|
|
b->n_top = MapNode(CONTENT_IGNORE);
|
|
|
|
|
|
|
|
if (getstringfield(L, index, "node_filler", nodename))
|
|
|
|
b->n_filler = MapNode(ndef->getId(nodename));
|
|
|
|
else
|
|
|
|
b->n_filler = b->n_top;
|
|
|
|
|
|
|
|
b->ntopnodes = getintfield_default(L, index, "num_top_nodes", 0);
|
|
|
|
|
|
|
|
b->height_min = getintfield_default(L, index, "height_min", 0);
|
|
|
|
b->height_max = getintfield_default(L, index, "height_max", 0);
|
|
|
|
b->heat_min = getfloatfield_default(L, index, "heat_min", 0.);
|
|
|
|
b->heat_max = getfloatfield_default(L, index, "heat_max", 0.);
|
|
|
|
b->humidity_min = getfloatfield_default(L, index, "humidity_min", 0.);
|
|
|
|
b->humidity_max = getfloatfield_default(L, index, "humidity_max", 0.);
|
|
|
|
|
|
|
|
b->np = new NoiseParams; // should read an entire NoiseParams later on...
|
|
|
|
getfloatfield(L, index, "scale", b->np->scale);
|
|
|
|
getfloatfield(L, index, "offset", b->np->offset);
|
|
|
|
|
|
|
|
b->groupid = (s8)groupid;
|
|
|
|
b->flags = 0; //reserved
|
|
|
|
|
|
|
|
bmgr->addBiome(b);
|
|
|
|
|
|
|
|
verbosestream << "register_biome: " << b->name << std::endl;
|
|
|
|
return 0;
|
2012-07-24 19:36:54 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-03-24 01:43:38 -04:00
|
|
|
static int l_register_ore(lua_State *L)
|
|
|
|
{
|
|
|
|
int index = 1;
|
|
|
|
luaL_checktype(L, index, LUA_TTABLE);
|
|
|
|
|
|
|
|
IWritableNodeDefManager *ndef = get_server(L)->getWritableNodeDefManager();
|
|
|
|
EmergeManager *emerge = get_server(L)->getEmergeManager();
|
|
|
|
|
|
|
|
enum OreType oretype = (OreType)getenumfield(L, index,
|
|
|
|
"ore_type", es_OreType, ORE_SCATTER);
|
|
|
|
Ore *ore = createOre(oretype);
|
2013-03-24 15:29:23 -04:00
|
|
|
if (!ore) {
|
|
|
|
errorstream << "register_ore: ore_type "
|
|
|
|
<< oretype << " not implemented";
|
|
|
|
return 0;
|
|
|
|
}
|
2013-03-24 01:43:38 -04:00
|
|
|
|
|
|
|
ore->ore_name = getstringfield_default(L, index, "ore", "");
|
|
|
|
ore->wherein_name = getstringfield_default(L, index, "wherein", "");
|
2013-03-24 15:29:23 -04:00
|
|
|
ore->clust_scarcity = getintfield_default(L, index, "clust_scarcity", 1);
|
|
|
|
ore->clust_num_ores = getintfield_default(L, index, "clust_num_ores", 1);
|
2013-03-24 01:43:38 -04:00
|
|
|
ore->clust_size = getintfield_default(L, index, "clust_size", 0);
|
|
|
|
ore->height_min = getintfield_default(L, index, "height_min", 0);
|
|
|
|
ore->height_max = getintfield_default(L, index, "height_max", 0);
|
|
|
|
ore->nthresh = getfloatfield_default(L, index, "noise_threshhold", 0.);
|
|
|
|
|
|
|
|
lua_getfield(L, index, "noise_params");
|
|
|
|
ore->np = read_noiseparams(L, -1);
|
|
|
|
lua_pop(L, 1);
|
|
|
|
|
|
|
|
ore->noise = NULL;
|
|
|
|
|
2013-03-24 15:29:23 -04:00
|
|
|
if (ore->clust_scarcity <= 0 || ore->clust_num_ores <= 0) {
|
|
|
|
errorstream << "register_ore: clust_scarcity and clust_num_ores"
|
|
|
|
"must be greater than 0";
|
|
|
|
delete ore;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2013-03-24 01:43:38 -04:00
|
|
|
emerge->ores.push_back(ore);
|
|
|
|
|
|
|
|
verbosestream << "register_ore: ore '" << ore->ore_name
|
|
|
|
<< "' registered" << std::endl;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2012-07-24 19:36:54 -04:00
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
// setting_set(name, value)
|
|
|
|
static int l_setting_set(lua_State *L)
|
|
|
|
{
|
|
|
|
const char *name = luaL_checkstring(L, 1);
|
|
|
|
const char *value = luaL_checkstring(L, 2);
|
|
|
|
g_settings->set(name, value);
|
|
|
|
return 0;
|
|
|
|
}
|
2012-07-24 19:36:54 -04:00
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
// setting_get(name)
|
|
|
|
static int l_setting_get(lua_State *L)
|
|
|
|
{
|
|
|
|
const char *name = luaL_checkstring(L, 1);
|
|
|
|
try{
|
|
|
|
std::string value = g_settings->get(name);
|
|
|
|
lua_pushstring(L, value.c_str());
|
|
|
|
} catch(SettingNotFoundException &e){
|
|
|
|
lua_pushnil(L);
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
2012-07-24 19:36:54 -04:00
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
// setting_getbool(name)
|
|
|
|
static int l_setting_getbool(lua_State *L)
|
|
|
|
{
|
|
|
|
const char *name = luaL_checkstring(L, 1);
|
|
|
|
try{
|
|
|
|
bool value = g_settings->getBool(name);
|
|
|
|
lua_pushboolean(L, value);
|
|
|
|
} catch(SettingNotFoundException &e){
|
|
|
|
lua_pushnil(L);
|
|
|
|
}
|
|
|
|
return 1;
|
2012-07-24 19:36:54 -04:00
|
|
|
}
|
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
// setting_save()
|
|
|
|
static int l_setting_save(lua_State *L)
|
2012-06-01 17:33:51 -04:00
|
|
|
{
|
2013-02-23 13:06:57 -05:00
|
|
|
get_server(L)->saveConfig();
|
|
|
|
return 0;
|
2012-06-01 17:33:51 -04:00
|
|
|
}
|
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
// chat_send_all(text)
|
|
|
|
static int l_chat_send_all(lua_State *L)
|
2012-06-01 17:33:51 -04:00
|
|
|
{
|
2013-02-23 13:06:57 -05:00
|
|
|
const char *text = luaL_checkstring(L, 1);
|
|
|
|
// Get server from registry
|
|
|
|
Server *server = get_server(L);
|
|
|
|
// Send
|
|
|
|
server->notifyPlayers(narrow_to_wide(text));
|
|
|
|
return 0;
|
2012-06-01 17:33:51 -04:00
|
|
|
}
|
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
// chat_send_player(name, text)
|
|
|
|
static int l_chat_send_player(lua_State *L)
|
2012-06-01 17:33:51 -04:00
|
|
|
{
|
2013-02-23 13:06:57 -05:00
|
|
|
const char *name = luaL_checkstring(L, 1);
|
|
|
|
const char *text = luaL_checkstring(L, 2);
|
|
|
|
// Get server from registry
|
|
|
|
Server *server = get_server(L);
|
|
|
|
// Send
|
|
|
|
server->notifyPlayer(name, narrow_to_wide(text));
|
|
|
|
return 0;
|
|
|
|
}
|
2012-06-01 17:33:51 -04:00
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
// get_player_privs(name, text)
|
|
|
|
static int l_get_player_privs(lua_State *L)
|
|
|
|
{
|
|
|
|
const char *name = luaL_checkstring(L, 1);
|
|
|
|
// Get server from registry
|
|
|
|
Server *server = get_server(L);
|
|
|
|
// Do it
|
|
|
|
lua_newtable(L);
|
|
|
|
int table = lua_gettop(L);
|
|
|
|
std::set<std::string> privs_s = server->getPlayerEffectivePrivs(name);
|
|
|
|
for(std::set<std::string>::const_iterator
|
|
|
|
i = privs_s.begin(); i != privs_s.end(); i++){
|
|
|
|
lua_pushboolean(L, true);
|
|
|
|
lua_setfield(L, table, i->c_str());
|
|
|
|
}
|
|
|
|
lua_pushvalue(L, table);
|
|
|
|
return 1;
|
|
|
|
}
|
2012-06-01 17:33:51 -04:00
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
// get_ban_list()
|
|
|
|
static int l_get_ban_list(lua_State *L)
|
|
|
|
{
|
|
|
|
lua_pushstring(L, get_server(L)->getBanDescription("").c_str());
|
|
|
|
return 1;
|
2012-07-24 19:36:54 -04:00
|
|
|
}
|
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
// get_ban_description()
|
|
|
|
static int l_get_ban_description(lua_State *L)
|
|
|
|
{
|
|
|
|
const char * ip_or_name = luaL_checkstring(L, 1);
|
|
|
|
lua_pushstring(L, get_server(L)->getBanDescription(std::string(ip_or_name)).c_str());
|
|
|
|
return 1;
|
|
|
|
}
|
2012-07-24 19:36:54 -04:00
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
// ban_player()
|
|
|
|
static int l_ban_player(lua_State *L)
|
2012-07-24 19:36:54 -04:00
|
|
|
{
|
2013-02-23 13:06:57 -05:00
|
|
|
const char * name = luaL_checkstring(L, 1);
|
|
|
|
Player *player = get_env(L)->getPlayer(name);
|
|
|
|
if(player == NULL)
|
2012-07-24 19:36:54 -04:00
|
|
|
{
|
2013-02-23 13:06:57 -05:00
|
|
|
lua_pushboolean(L, false); // no such player
|
|
|
|
return 1;
|
2012-07-24 19:36:54 -04:00
|
|
|
}
|
2013-02-23 13:06:57 -05:00
|
|
|
try
|
2012-07-24 19:36:54 -04:00
|
|
|
{
|
2013-02-23 13:06:57 -05:00
|
|
|
Address addr = get_server(L)->getPeerAddress(get_env(L)->getPlayer(name)->peer_id);
|
|
|
|
std::string ip_str = addr.serializeString();
|
|
|
|
get_server(L)->setIpBanned(ip_str, name);
|
2012-07-24 19:36:54 -04:00
|
|
|
}
|
2013-02-23 13:06:57 -05:00
|
|
|
catch(con::PeerNotFoundException) // unlikely
|
2012-07-24 19:36:54 -04:00
|
|
|
{
|
2013-02-23 13:06:57 -05:00
|
|
|
dstream << __FUNCTION_NAME << ": peer was not found" << std::endl;
|
|
|
|
lua_pushboolean(L, false); // error
|
|
|
|
return 1;
|
2012-07-24 19:36:54 -04:00
|
|
|
}
|
2013-02-23 13:06:57 -05:00
|
|
|
lua_pushboolean(L, true);
|
|
|
|
return 1;
|
2012-07-24 19:36:54 -04:00
|
|
|
}
|
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
// unban_player_or_ip()
|
|
|
|
static int l_unban_player_of_ip(lua_State *L)
|
2012-07-24 19:36:54 -04:00
|
|
|
{
|
2013-02-23 13:06:57 -05:00
|
|
|
const char * ip_or_name = luaL_checkstring(L, 1);
|
|
|
|
get_server(L)->unsetIpBanned(ip_or_name);
|
|
|
|
lua_pushboolean(L, true);
|
|
|
|
return 1;
|
2012-07-24 19:36:54 -04:00
|
|
|
}
|
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
// show_formspec(playername,formname,formspec)
|
|
|
|
static int l_show_formspec(lua_State *L)
|
2012-07-24 19:36:54 -04:00
|
|
|
{
|
2013-02-23 13:06:57 -05:00
|
|
|
const char *playername = luaL_checkstring(L, 1);
|
|
|
|
const char *formname = luaL_checkstring(L, 2);
|
|
|
|
const char *formspec = luaL_checkstring(L, 3);
|
2012-07-24 19:36:54 -04:00
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
if(get_server(L)->showFormspec(playername,formspec,formname))
|
|
|
|
{
|
|
|
|
lua_pushboolean(L, true);
|
|
|
|
}else{
|
|
|
|
lua_pushboolean(L, false);
|
|
|
|
}
|
|
|
|
return 1;
|
2012-07-24 19:36:54 -04:00
|
|
|
}
|
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
// get_dig_params(groups, tool_capabilities[, time_from_last_punch])
|
|
|
|
static int l_get_dig_params(lua_State *L)
|
2012-07-24 19:36:54 -04:00
|
|
|
{
|
2013-02-23 13:06:57 -05:00
|
|
|
std::map<std::string, int> groups;
|
|
|
|
read_groups(L, 1, groups);
|
|
|
|
ToolCapabilities tp = read_tool_capabilities(L, 2);
|
|
|
|
if(lua_isnoneornil(L, 3))
|
|
|
|
push_dig_params(L, getDigParams(groups, &tp));
|
|
|
|
else
|
|
|
|
push_dig_params(L, getDigParams(groups, &tp,
|
|
|
|
luaL_checknumber(L, 3)));
|
|
|
|
return 1;
|
2012-07-24 19:36:54 -04:00
|
|
|
}
|
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
// get_hit_params(groups, tool_capabilities[, time_from_last_punch])
|
|
|
|
static int l_get_hit_params(lua_State *L)
|
2012-07-24 19:36:54 -04:00
|
|
|
{
|
2013-02-23 13:06:57 -05:00
|
|
|
std::map<std::string, int> groups;
|
|
|
|
read_groups(L, 1, groups);
|
|
|
|
ToolCapabilities tp = read_tool_capabilities(L, 2);
|
|
|
|
if(lua_isnoneornil(L, 3))
|
|
|
|
push_hit_params(L, getHitParams(groups, &tp));
|
|
|
|
else
|
|
|
|
push_hit_params(L, getHitParams(groups, &tp,
|
|
|
|
luaL_checknumber(L, 3)));
|
|
|
|
return 1;
|
|
|
|
}
|
2012-07-24 19:36:54 -04:00
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
// get_current_modname()
|
|
|
|
static int l_get_current_modname(lua_State *L)
|
|
|
|
{
|
|
|
|
lua_getfield(L, LUA_REGISTRYINDEX, "minetest_current_modname");
|
|
|
|
return 1;
|
|
|
|
}
|
2012-07-24 19:36:54 -04:00
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
// get_modpath(modname)
|
|
|
|
static int l_get_modpath(lua_State *L)
|
|
|
|
{
|
|
|
|
std::string modname = luaL_checkstring(L, 1);
|
|
|
|
// Do it
|
|
|
|
if(modname == "__builtin"){
|
|
|
|
std::string path = get_server(L)->getBuiltinLuaPath();
|
|
|
|
lua_pushstring(L, path.c_str());
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
const ModSpec *mod = get_server(L)->getModSpec(modname);
|
|
|
|
if(!mod){
|
|
|
|
lua_pushnil(L);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
lua_pushstring(L, mod->path.c_str());
|
|
|
|
return 1;
|
2012-07-24 19:36:54 -04:00
|
|
|
}
|
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
// get_modnames()
|
|
|
|
// the returned list is sorted alphabetically for you
|
|
|
|
static int l_get_modnames(lua_State *L)
|
2012-07-24 19:36:54 -04:00
|
|
|
{
|
2013-02-23 13:06:57 -05:00
|
|
|
// Get a list of mods
|
2012-12-20 12:19:49 -05:00
|
|
|
std::list<std::string> mods_unsorted, mods_sorted;
|
2013-02-23 13:06:57 -05:00
|
|
|
get_server(L)->getModNames(mods_unsorted);
|
2012-07-24 19:36:54 -04:00
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
// Take unsorted items from mods_unsorted and sort them into
|
|
|
|
// mods_sorted; not great performance but the number of mods on a
|
|
|
|
// server will likely be small.
|
2012-12-20 12:19:49 -05:00
|
|
|
for(std::list<std::string>::iterator i = mods_unsorted.begin();
|
|
|
|
i != mods_unsorted.end(); ++i)
|
2013-02-23 13:06:57 -05:00
|
|
|
{
|
|
|
|
bool added = false;
|
2012-12-20 12:19:49 -05:00
|
|
|
for(std::list<std::string>::iterator x = mods_sorted.begin();
|
2013-03-13 20:24:07 -04:00
|
|
|
x != mods_sorted.end(); ++x)
|
2013-02-23 13:06:57 -05:00
|
|
|
{
|
|
|
|
// I doubt anybody using Minetest will be using
|
|
|
|
// anything not ASCII based :)
|
|
|
|
if((*i).compare(*x) <= 0)
|
|
|
|
{
|
2012-12-20 12:19:49 -05:00
|
|
|
mods_sorted.insert(x, *i);
|
2013-02-23 13:06:57 -05:00
|
|
|
added = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if(!added)
|
|
|
|
mods_sorted.push_back(*i);
|
|
|
|
}
|
2012-07-24 19:36:54 -04:00
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
// Get the table insertion function from Lua.
|
|
|
|
lua_getglobal(L, "table");
|
|
|
|
lua_getfield(L, -1, "insert");
|
|
|
|
int insertion_func = lua_gettop(L);
|
|
|
|
|
|
|
|
// Package them up for Lua
|
|
|
|
lua_newtable(L);
|
|
|
|
int new_table = lua_gettop(L);
|
2012-12-20 12:19:49 -05:00
|
|
|
std::list<std::string>::iterator i = mods_sorted.begin();
|
2013-02-23 13:06:57 -05:00
|
|
|
while(i != mods_sorted.end())
|
|
|
|
{
|
|
|
|
lua_pushvalue(L, insertion_func);
|
|
|
|
lua_pushvalue(L, new_table);
|
|
|
|
lua_pushstring(L, (*i).c_str());
|
|
|
|
if(lua_pcall(L, 2, 0, 0) != 0)
|
|
|
|
{
|
|
|
|
script_error(L, "error: %s", lua_tostring(L, -1));
|
|
|
|
}
|
2012-12-20 12:19:49 -05:00
|
|
|
++i;
|
2013-02-23 13:06:57 -05:00
|
|
|
}
|
|
|
|
return 1;
|
2012-07-24 19:36:54 -04:00
|
|
|
}
|
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
// get_worldpath()
|
|
|
|
static int l_get_worldpath(lua_State *L)
|
2012-07-24 19:36:54 -04:00
|
|
|
{
|
2013-02-23 13:06:57 -05:00
|
|
|
std::string worldpath = get_server(L)->getWorldPath();
|
|
|
|
lua_pushstring(L, worldpath.c_str());
|
|
|
|
return 1;
|
2012-06-01 17:33:51 -04:00
|
|
|
}
|
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
// sound_play(spec, parameters)
|
|
|
|
static int l_sound_play(lua_State *L)
|
2011-11-21 04:15:15 -05:00
|
|
|
{
|
2013-02-23 13:06:57 -05:00
|
|
|
SimpleSoundSpec spec;
|
|
|
|
read_soundspec(L, 1, spec);
|
|
|
|
ServerSoundParams params;
|
|
|
|
read_server_sound_params(L, 2, params);
|
|
|
|
s32 handle = get_server(L)->playSound(spec, params);
|
|
|
|
lua_pushinteger(L, handle);
|
|
|
|
return 1;
|
2011-11-21 04:15:15 -05:00
|
|
|
}
|
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
// sound_stop(handle)
|
|
|
|
static int l_sound_stop(lua_State *L)
|
2011-11-26 08:19:03 -05:00
|
|
|
{
|
2013-02-23 13:06:57 -05:00
|
|
|
int handle = luaL_checkinteger(L, 1);
|
|
|
|
get_server(L)->stopSound(handle);
|
|
|
|
return 0;
|
2011-11-26 08:19:03 -05:00
|
|
|
}
|
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
// is_singleplayer()
|
|
|
|
static int l_is_singleplayer(lua_State *L)
|
2011-11-11 12:33:17 -05:00
|
|
|
{
|
2013-02-23 13:06:57 -05:00
|
|
|
lua_pushboolean(L, get_server(L)->isSingleplayer());
|
|
|
|
return 1;
|
2012-03-09 18:38:48 -05:00
|
|
|
}
|
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
// get_password_hash(name, raw_password)
|
|
|
|
static int l_get_password_hash(lua_State *L)
|
2012-03-09 18:38:48 -05:00
|
|
|
{
|
2013-02-23 13:06:57 -05:00
|
|
|
std::string name = luaL_checkstring(L, 1);
|
|
|
|
std::string raw_password = luaL_checkstring(L, 2);
|
|
|
|
std::string hash = translatePassword(name,
|
|
|
|
narrow_to_wide(raw_password));
|
|
|
|
lua_pushstring(L, hash.c_str());
|
|
|
|
return 1;
|
2011-11-11 12:33:17 -05:00
|
|
|
}
|
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
// notify_authentication_modified(name)
|
|
|
|
static int l_notify_authentication_modified(lua_State *L)
|
2011-11-11 12:33:17 -05:00
|
|
|
{
|
2013-02-23 13:06:57 -05:00
|
|
|
std::string name = "";
|
|
|
|
if(lua_isstring(L, 1))
|
|
|
|
name = lua_tostring(L, 1);
|
|
|
|
get_server(L)->reportPrivsModified(name);
|
|
|
|
return 0;
|
2011-11-11 12:33:17 -05:00
|
|
|
}
|
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
// rollback_get_last_node_actor(p, range, seconds) -> actor, p, seconds
|
|
|
|
static int l_rollback_get_last_node_actor(lua_State *L)
|
2011-11-11 12:33:17 -05:00
|
|
|
{
|
2013-02-23 13:06:57 -05:00
|
|
|
v3s16 p = read_v3s16(L, 1);
|
|
|
|
int range = luaL_checknumber(L, 2);
|
|
|
|
int seconds = luaL_checknumber(L, 3);
|
|
|
|
Server *server = get_server(L);
|
|
|
|
IRollbackManager *rollback = server->getRollbackManager();
|
|
|
|
v3s16 act_p;
|
|
|
|
int act_seconds = 0;
|
|
|
|
std::string actor = rollback->getLastNodeActor(p, range, seconds, &act_p, &act_seconds);
|
|
|
|
lua_pushstring(L, actor.c_str());
|
|
|
|
push_v3s16(L, act_p);
|
|
|
|
lua_pushnumber(L, act_seconds);
|
|
|
|
return 3;
|
2011-11-11 12:33:17 -05:00
|
|
|
}
|
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
// rollback_revert_actions_by(actor, seconds) -> bool, log messages
|
|
|
|
static int l_rollback_revert_actions_by(lua_State *L)
|
2011-11-11 20:21:40 -05:00
|
|
|
{
|
2013-02-23 13:06:57 -05:00
|
|
|
std::string actor = luaL_checkstring(L, 1);
|
|
|
|
int seconds = luaL_checknumber(L, 2);
|
|
|
|
Server *server = get_server(L);
|
|
|
|
IRollbackManager *rollback = server->getRollbackManager();
|
|
|
|
std::list<RollbackAction> actions = rollback->getRevertActions(actor, seconds);
|
|
|
|
std::list<std::string> log;
|
|
|
|
bool success = server->rollbackRevertActions(actions, &log);
|
|
|
|
// Push boolean result
|
|
|
|
lua_pushboolean(L, success);
|
|
|
|
// Get the table insert function and push the log table
|
|
|
|
lua_getglobal(L, "table");
|
|
|
|
lua_getfield(L, -1, "insert");
|
|
|
|
int table_insert = lua_gettop(L);
|
|
|
|
lua_newtable(L);
|
|
|
|
int table = lua_gettop(L);
|
|
|
|
for(std::list<std::string>::const_iterator i = log.begin();
|
|
|
|
i != log.end(); i++)
|
|
|
|
{
|
|
|
|
lua_pushvalue(L, table_insert);
|
|
|
|
lua_pushvalue(L, table);
|
|
|
|
lua_pushstring(L, i->c_str());
|
|
|
|
if(lua_pcall(L, 2, 0, 0))
|
|
|
|
script_error(L, "error: %s", lua_tostring(L, -1));
|
|
|
|
}
|
|
|
|
lua_remove(L, -2); // Remove table
|
|
|
|
lua_remove(L, -2); // Remove insert
|
|
|
|
return 2;
|
|
|
|
}
|
2012-10-23 17:11:24 -04:00
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
static const struct luaL_Reg minetest_f [] = {
|
|
|
|
{"debug", l_debug},
|
|
|
|
{"log", l_log},
|
|
|
|
{"request_shutdown", l_request_shutdown},
|
|
|
|
{"get_server_status", l_get_server_status},
|
|
|
|
{"register_item_raw", l_register_item_raw},
|
|
|
|
{"register_alias_raw", l_register_alias_raw},
|
|
|
|
{"register_craft", l_register_craft},
|
|
|
|
{"register_biome", l_register_biome},
|
|
|
|
{"register_biome_groups", l_register_biome_groups},
|
2013-03-24 01:43:38 -04:00
|
|
|
{"register_ore", l_register_ore},
|
2013-02-23 13:06:57 -05:00
|
|
|
{"setting_set", l_setting_set},
|
|
|
|
{"setting_get", l_setting_get},
|
|
|
|
{"setting_getbool", l_setting_getbool},
|
|
|
|
{"setting_save",l_setting_save},
|
|
|
|
{"chat_send_all", l_chat_send_all},
|
|
|
|
{"chat_send_player", l_chat_send_player},
|
|
|
|
{"get_player_privs", l_get_player_privs},
|
|
|
|
{"get_ban_list", l_get_ban_list},
|
|
|
|
{"get_ban_description", l_get_ban_description},
|
|
|
|
{"ban_player", l_ban_player},
|
|
|
|
{"unban_player_or_ip", l_unban_player_of_ip},
|
|
|
|
{"get_inventory", l_get_inventory},
|
|
|
|
{"create_detached_inventory_raw", l_create_detached_inventory_raw},
|
|
|
|
{"show_formspec", l_show_formspec},
|
|
|
|
{"get_dig_params", l_get_dig_params},
|
|
|
|
{"get_hit_params", l_get_hit_params},
|
|
|
|
{"get_current_modname", l_get_current_modname},
|
|
|
|
{"get_modpath", l_get_modpath},
|
|
|
|
{"get_modnames", l_get_modnames},
|
|
|
|
{"get_worldpath", l_get_worldpath},
|
|
|
|
{"sound_play", l_sound_play},
|
|
|
|
{"sound_stop", l_sound_stop},
|
|
|
|
{"is_singleplayer", l_is_singleplayer},
|
|
|
|
{"get_password_hash", l_get_password_hash},
|
|
|
|
{"notify_authentication_modified", l_notify_authentication_modified},
|
|
|
|
{"get_craft_result", l_get_craft_result},
|
|
|
|
{"get_craft_recipe", l_get_craft_recipe},
|
2013-03-03 19:55:16 -05:00
|
|
|
{"get_all_craft_recipes", l_get_all_craft_recipes},
|
2013-02-23 13:06:57 -05:00
|
|
|
{"rollback_get_last_node_actor", l_rollback_get_last_node_actor},
|
|
|
|
{"rollback_revert_actions_by", l_rollback_revert_actions_by},
|
2013-01-23 12:32:02 -05:00
|
|
|
{"add_particle", l_add_particle},
|
|
|
|
{"add_particlespawner", l_add_particlespawner},
|
|
|
|
{"delete_particlespawner", l_delete_particlespawner},
|
2013-02-23 13:06:57 -05:00
|
|
|
{NULL, NULL}
|
|
|
|
};
|
2012-12-18 13:23:16 -05:00
|
|
|
|
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
/*
|
|
|
|
Main export function
|
|
|
|
*/
|
2011-11-11 20:21:40 -05:00
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
void scriptapi_export(lua_State *L, Server *server)
|
2011-11-11 12:33:17 -05:00
|
|
|
{
|
|
|
|
realitycheck(L);
|
|
|
|
assert(lua_checkstack(L, 20));
|
2013-02-23 13:06:57 -05:00
|
|
|
verbosestream<<"scriptapi_export()"<<std::endl;
|
2011-11-11 19:25:30 -05:00
|
|
|
StackUnroller stack_unroller(L);
|
2011-11-11 12:33:17 -05:00
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
// Store server as light userdata in registry
|
|
|
|
lua_pushlightuserdata(L, server);
|
|
|
|
lua_setfield(L, LUA_REGISTRYINDEX, "minetest_server");
|
2011-11-11 12:33:17 -05:00
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
// Register global functions in table minetest
|
|
|
|
lua_newtable(L);
|
|
|
|
luaL_register(L, NULL, minetest_f);
|
|
|
|
lua_setglobal(L, "minetest");
|
2011-11-11 12:33:17 -05:00
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
// Get the main minetest table
|
|
|
|
lua_getglobal(L, "minetest");
|
2011-11-12 10:37:14 -05:00
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
// Add tables to minetest
|
|
|
|
lua_newtable(L);
|
|
|
|
lua_setfield(L, -2, "object_refs");
|
|
|
|
lua_newtable(L);
|
|
|
|
lua_setfield(L, -2, "luaentities");
|
2011-11-12 10:37:14 -05:00
|
|
|
|
2013-02-23 13:06:57 -05:00
|
|
|
// Register wrappers
|
|
|
|
LuaItemStack::Register(L);
|
|
|
|
InvRef::Register(L);
|
|
|
|
NodeMetaRef::Register(L);
|
|
|
|
NodeTimerRef::Register(L);
|
|
|
|
ObjectRef::Register(L);
|
|
|
|
EnvRef::Register(L);
|
|
|
|
LuaPseudoRandom::Register(L);
|
|
|
|
LuaPerlinNoise::Register(L);
|
|
|
|
LuaPerlinNoiseMap::Register(L);
|
2011-11-11 12:33:17 -05:00
|
|
|
}
|