mirror of
https://github.com/moparisthebest/hexchat
synced 2024-11-02 08:25:04 -04:00
ec301a5a54
On Unix leave $(libdir)/hexchat/plugins for plugin packagers, on Windows prevent users from modifying Program Files by ignoring everything except bundled plugins
1887 lines
46 KiB
C
1887 lines
46 KiB
C
/*
|
|
* X-Chat 2.0 LUA Plugin
|
|
*
|
|
* Copyright (c) 2007 Hanno Hecker
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; version 2 of the License.
|
|
*
|
|
* 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 General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA
|
|
*/
|
|
/*
|
|
* $Id: lua.c 91 2007-06-09 18:44:03Z vetinari $
|
|
* $Revision: 91 $
|
|
* $Date: 2007-06-09 20:44:03 +0200 (Szo, 09 jún. 2007) $
|
|
*/
|
|
/*
|
|
* TODO:
|
|
* * compile (was OK)/run on IRIX
|
|
* ? localize error msgs? ... maybe later
|
|
* ? make xchat.print() like print() which does an tostring() on
|
|
* everything it gets?
|
|
* ? add /LUA -s <code>? ... add a new script from cmdline... this state
|
|
* is not removed after the pcall(), but prints a name, which may
|
|
* be used to unload this virtual script. ... no xchat_register(),
|
|
* xchat_init() should be needed
|
|
* ... don't disable xchat.hook_* for this
|
|
* ? timer name per state/script and not per plugin?
|
|
*/
|
|
#define LXC_NAME "Lua"
|
|
#define LXC_DESC "Lua scripting interface"
|
|
#define LXC_VERSION "0.7 (r91)"
|
|
|
|
#include <stdlib.h>
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <sys/types.h>
|
|
#include <sys/stat.h>
|
|
#include <fcntl.h>
|
|
#include <errno.h>
|
|
#include <ctype.h>
|
|
|
|
#ifdef _WIN32
|
|
#include <direct.h> /* for getcwd */
|
|
#include "../../src/dirent/dirent-win32.h"
|
|
#endif
|
|
|
|
#if !( defined(_WIN32) || defined(LXC_XCHAT_GETTEXT) )
|
|
# include <libintl.h>
|
|
#endif
|
|
|
|
#ifndef PATH_MAX /* hurd */
|
|
# define PATH_MAX 1024
|
|
#endif
|
|
|
|
#include <lua.h>
|
|
#include <lauxlib.h>
|
|
#include <lualib.h>
|
|
|
|
#define lua_pop(L,n) lua_settop(L, -(n)-1)
|
|
|
|
#include "xchat-plugin.h"
|
|
|
|
static xchat_plugin *ph; /* plugin handle */
|
|
|
|
#define LXC_STRIP_COLOR 1
|
|
#define LXC_STRIP_ATTR 2
|
|
#define LXC_STRIP_ALL (LXC_STRIP_COLOR|LXC_STRIP_ATTR)
|
|
|
|
/* registered hooks */
|
|
struct lxc_hooks {
|
|
const char *name;
|
|
xchat_hook *hook;
|
|
struct lxc_hooks *next;
|
|
};
|
|
|
|
/* single linked list of all lua states^Wscripts ;-) */
|
|
struct lxc_States {
|
|
lua_State *state; /* the lua state of the script */
|
|
char file[PATH_MAX+1]; /* the file name of the script */
|
|
struct lxc_hooks *hooks; /* all hooks this script registered */
|
|
void *gui; /* the gui entry in windows->plugins and scripts... */
|
|
struct lxc_States *next;
|
|
};
|
|
|
|
static struct lxc_States *lxc_states = NULL;
|
|
|
|
/* user/script supplied data for a callback */
|
|
struct lxc_userdata {
|
|
int idx; /* table index */
|
|
int type; /* lua type: */
|
|
const char *string; /* only strings, ... */
|
|
double num; /* numbers and booleans are supported */
|
|
struct lxc_userdata *next;
|
|
};
|
|
|
|
/* callback data */
|
|
struct lxc_cbdata {
|
|
lua_State *state;
|
|
const char *func;
|
|
xchat_hook *hook; /* timer ... */
|
|
struct lxc_userdata *data;
|
|
};
|
|
|
|
static char lxc_event_name[1024] = "\0";
|
|
|
|
static int lxc_run_hook(char *word[], char *word_eol[], void *data);
|
|
static int lxc_run_print(char *word[], void *data);
|
|
static int lxc_run_timer(void *data);
|
|
|
|
static int lxc_hook_command(lua_State *L);
|
|
static int lxc_hook_server(lua_State *L);
|
|
static int lxc_hook_print(lua_State *L);
|
|
static int lxc_event(lua_State *L);
|
|
static int lxc_hook_timer(lua_State *L);
|
|
static int lxc_unhook(lua_State *L);
|
|
|
|
static int lxc_command(lua_State *L);
|
|
static int lxc_print(lua_State *L);
|
|
static int lxc_emit_print(lua_State *L);
|
|
static int lxc_send_modes(lua_State *L);
|
|
static int lxc_find_context(lua_State *L);
|
|
static int lxc_get_context(lua_State *L);
|
|
static int lxc_get_info(lua_State *L);
|
|
static int lxc_get_prefs(lua_State *L);
|
|
static int lxc_set_context(lua_State *L);
|
|
static int lxc_nickcmp(lua_State *L);
|
|
|
|
static int lxc_list_get(lua_State *L);
|
|
static int lxc_list_fields(lua_State *L);
|
|
static int lxc_gettext(lua_State *L);
|
|
|
|
static int lxc_bits(lua_State *L);
|
|
|
|
static luaL_reg lxc_functions[] = {
|
|
{"hook_command", lxc_hook_command },
|
|
/* TODO:
|
|
{"hook_fd", lxc_hook_fd },
|
|
*/
|
|
{"hook_print", lxc_hook_print },
|
|
{"hook_server", lxc_hook_server },
|
|
{"hook_timer", lxc_hook_timer },
|
|
{"unhook", lxc_unhook },
|
|
|
|
{"event", lxc_event },
|
|
|
|
{"command", lxc_command },
|
|
{"print", lxc_print },
|
|
{"emit_print", lxc_emit_print },
|
|
{"send_modes", lxc_send_modes },
|
|
{"find_context", lxc_find_context },
|
|
{"get_context", lxc_get_context },
|
|
{"get_info", lxc_get_info },
|
|
{"get_prefs", lxc_get_prefs },
|
|
{"set_context", lxc_set_context },
|
|
|
|
{"nickcmp", lxc_nickcmp },
|
|
|
|
{"list_get", lxc_list_get },
|
|
{"list_fields", lxc_list_fields },
|
|
|
|
{"gettext", lxc_gettext},
|
|
/* helper function for bit flags */
|
|
{"bits", lxc_bits },
|
|
{NULL, NULL}
|
|
};
|
|
|
|
static struct {
|
|
const char *name;
|
|
long value;
|
|
} lxc_consts[] = {
|
|
{"EAT_NONE", XCHAT_EAT_NONE},
|
|
{"EAT_XCHAT", XCHAT_EAT_XCHAT},
|
|
{"EAT_PLUGIN", XCHAT_EAT_PLUGIN},
|
|
{"EAT_ALL", XCHAT_EAT_ALL},
|
|
|
|
/* unused until hook_fd is done
|
|
{"FD_READ", XCHAT_FD_READ},
|
|
{"FD_WRITE", XCHAT_FD_WRITE},
|
|
{"FD_EXCEPTION", XCHAT_FD_EXCEPTION},
|
|
{"FD_NOTSOCKET", XCHAT_FD_NOTSOCKET},
|
|
*/
|
|
|
|
{"PRI_HIGHEST", XCHAT_PRI_HIGHEST},
|
|
{"PRI_HIGH", XCHAT_PRI_HIGH},
|
|
{"PRI_NORM", XCHAT_PRI_NORM},
|
|
{"PRI_LOW", XCHAT_PRI_LOW},
|
|
{"PRI_LOWEST", XCHAT_PRI_LOWEST},
|
|
|
|
/* for: clean = xchat.strip(dirty, xchat.STRIP_ALL) */
|
|
{"STRIP_COLOR", LXC_STRIP_COLOR},
|
|
{"STRIP_ATTR", LXC_STRIP_ATTR},
|
|
{"STRIP_ALL", LXC_STRIP_ALL},
|
|
|
|
/* for xchat.commandf("GUI COLOR %d", xchat.TAB_HILIGHT) */
|
|
{"TAB_DEFAULT", 0},
|
|
{"TAB_NEWDATA", 1},
|
|
{"TAB_NEWMSG", 2},
|
|
{"TAB_HILIGHT", 3},
|
|
|
|
{NULL, 0}
|
|
};
|
|
|
|
|
|
#ifdef DEBUG
|
|
static void stackDump (lua_State *L, const char *msg) {
|
|
int i, t;
|
|
int top = lua_gettop(L);
|
|
|
|
fprintf(stderr, "%s\n", msg);
|
|
for (i = 1; i <= top; i++) { /* repeat for each level */
|
|
t = lua_type(L, i);
|
|
switch (t) {
|
|
|
|
case LUA_TSTRING: /* strings */
|
|
fprintf(stderr, "`%s'", lua_tostring(L, i));
|
|
break;
|
|
|
|
case LUA_TBOOLEAN: /* booleans */
|
|
fprintf(stderr, lua_toboolean(L, i) ? "true" : "false");
|
|
break;
|
|
|
|
case LUA_TNUMBER: /* numbers */
|
|
fprintf(stderr, "%g", lua_tonumber(L, i));
|
|
break;
|
|
|
|
default: /* other values */
|
|
fprintf(stderr, "%s", lua_typename(L, t));
|
|
break;
|
|
|
|
}
|
|
fprintf(stderr, " "); /* put a separator */
|
|
}
|
|
fprintf(stderr, "\n"); /* end the listing */
|
|
}
|
|
#endif /* DEBUG */
|
|
|
|
static int lxc__newindex(lua_State *L)
|
|
{
|
|
int i;
|
|
const char *name = lua_tostring(L, 2);
|
|
|
|
luaL_getmetatable(L, "xchat"); /* 4 */
|
|
|
|
lua_pushnil(L); /* 5 */
|
|
while (lua_next(L, 4) != 0) {
|
|
if ((lua_type(L, -2) == LUA_TSTRING)
|
|
&& strcmp("__index", lua_tostring(L, -2)) == 0)
|
|
break; /* now __index is 5, table 6 */
|
|
lua_pop(L, 1);
|
|
}
|
|
|
|
lua_pushnil(L);
|
|
while (lua_next(L, 6) != 0) {
|
|
if ((lua_type(L, -2) == LUA_TSTRING)
|
|
&& strcmp(name, lua_tostring(L, -2)) == 0) {
|
|
for (i=0; lxc_consts[i].name; i++) {
|
|
if (strcmp(name, lxc_consts[i].name) == 0) {
|
|
luaL_error(L,
|
|
"`xchat.%s' is a readonly constant", lua_tostring(L, 2));
|
|
return 0;
|
|
}
|
|
}
|
|
}
|
|
lua_pop(L, 1);
|
|
}
|
|
|
|
lua_pushvalue(L, 2);
|
|
lua_pushvalue(L, 3);
|
|
lua_rawset(L, 6);
|
|
|
|
lua_settop(L, 1);
|
|
return 0;
|
|
}
|
|
|
|
static int luaopen_xchat(lua_State *L)
|
|
{
|
|
int i;
|
|
/*
|
|
* wrappers for xchat.printf() and xchat.commandf()
|
|
* ... xchat.strip
|
|
*/
|
|
#define LXC_WRAPPERS "function xchat.printf(...)\n" \
|
|
" xchat.print(string.format(unpack(arg)))\n" \
|
|
"end\n" \
|
|
"function xchat.commandf(...)\n" \
|
|
" xchat.command(string.format(unpack(arg)))\n" \
|
|
"end\n" \
|
|
"function xchat.strip(str, flags)\n" \
|
|
" if flags == nil then\n" \
|
|
" flags = xchat.STRIP_ALL\n" \
|
|
" end\n" \
|
|
" local bits = xchat.bits(flags)\n" \
|
|
" if bits[1] then\n" \
|
|
" str = string.gsub(\n" \
|
|
" string.gsub(str, \"\\3%d%d?,%d%d?\", \"\"),\n" \
|
|
" \"\\3%d%d?\", \"\")\n" \
|
|
" end\n" \
|
|
" if bits[2] then\n" \
|
|
" -- bold, beep, reset, reverse, underline\n" \
|
|
" str = string.gsub(str,\n" \
|
|
" \"[\\2\\7\\15\\22\\31]\", \"\")\n" \
|
|
" end\n" \
|
|
" return str\n" \
|
|
"end\n"
|
|
|
|
#if defined(LUA_VERSION_NUM) && (LUA_VERSION_NUM >= 501)
|
|
luaL_register(L, "xchat", lxc_functions);
|
|
(void)luaL_dostring(L, LXC_WRAPPERS);
|
|
#else
|
|
luaL_openlib(L, "xchat", lxc_functions, 0);
|
|
lua_dostring(L, LXC_WRAPPERS);
|
|
#endif
|
|
|
|
luaL_newmetatable(L, "xchat");
|
|
|
|
lua_pushliteral(L, "__index");
|
|
lua_newtable(L);
|
|
|
|
lua_pushstring(L, "ARCH");
|
|
#ifdef _WIN32
|
|
lua_pushstring(L, "Windows");
|
|
#else
|
|
lua_pushstring(L, "Unix");
|
|
#endif
|
|
lua_settable(L, -3); /* add to table __index */
|
|
|
|
for (i=0; lxc_consts[i].name; i++) {
|
|
lua_pushstring(L, lxc_consts[i].name);
|
|
lua_pushnumber(L, lxc_consts[i].value);
|
|
lua_settable(L, -3); /* add to table __index */
|
|
}
|
|
lua_settable(L, -3); /* add to metatable */
|
|
|
|
lua_pushliteral(L, "__newindex");
|
|
lua_pushcfunction(L, lxc__newindex);
|
|
lua_settable(L, -3);
|
|
/*
|
|
lua_pushliteral(L, "__metatable");
|
|
lua_pushstring(L, "nothing to see here, move along");
|
|
lua_settable(L, -3);
|
|
*/
|
|
lua_setmetatable(L, -2);
|
|
lua_pop(L, 1);
|
|
return 1;
|
|
}
|
|
|
|
lua_State *lxc_new_state()
|
|
{
|
|
#if defined(LUA_VERSION_NUM) && (LUA_VERSION_NUM >= 501)
|
|
lua_State *L = luaL_newstate(); /* opens Lua */
|
|
luaL_openlibs(L);
|
|
#else
|
|
lua_State *L = lua_open(); /* opens Lua */
|
|
luaopen_base(L); /* opens the basic library */
|
|
luaopen_table(L); /* opens the table library */
|
|
luaopen_io(L); /* opens the I/O library */
|
|
luaopen_string(L); /* opens the string lib. */
|
|
luaopen_math(L); /* opens the math lib. */
|
|
#endif
|
|
|
|
luaopen_xchat(L);
|
|
return L;
|
|
}
|
|
|
|
static int
|
|
lxc_load_file(const char *script)
|
|
{
|
|
lua_State *L;
|
|
struct lxc_States *state; /* pointer to lua states list */
|
|
struct lxc_States *st; /* pointer to lua states list */
|
|
|
|
L = lxc_new_state();
|
|
state = malloc(sizeof(struct lxc_States));
|
|
if (state == NULL) {
|
|
xchat_printf(ph, "malloc() failed: %s\n", strerror(errno));
|
|
lua_close(L);
|
|
return 0;
|
|
}
|
|
|
|
state->state = L;
|
|
snprintf(state->file, PATH_MAX, script);
|
|
state->next = NULL;
|
|
state->hooks = NULL;
|
|
state->gui = NULL;
|
|
|
|
if (luaL_loadfile(L, script) || lua_pcall(L, 0, 0, 0)) {
|
|
xchat_printf(ph, "Lua plugin: error loading script %s",
|
|
lua_tostring(L, -1));
|
|
lua_close(L);
|
|
free(state);
|
|
return 0;
|
|
}
|
|
|
|
if (!lxc_states)
|
|
lxc_states = state;
|
|
else {
|
|
st = lxc_states;
|
|
while (st->next)
|
|
st = st->next;
|
|
st->next = state;
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
static void
|
|
lxc_autoload_from_path(const char *path)
|
|
{
|
|
DIR *dir;
|
|
struct dirent *ent;
|
|
char *file;
|
|
int len;
|
|
/* xchat_printf(ph, "loading from %s\n", path); */
|
|
dir = opendir(path);
|
|
if (dir) {
|
|
while ((ent = readdir(dir))) {
|
|
len = strlen(ent->d_name);
|
|
if (len > 4 && strcasecmp(".lua", ent->d_name + len - 4) == 0) {
|
|
file = malloc(len + strlen(path) + 2);
|
|
if (file == NULL) {
|
|
xchat_printf(ph, "lxc_autoload_from_path(): malloc failed: %s",
|
|
strerror(errno));
|
|
break;
|
|
}
|
|
sprintf(file, "%s/%s", path, ent->d_name);
|
|
(void)lxc_load_file((const char *)file);
|
|
free(file);
|
|
}
|
|
}
|
|
closedir(dir);
|
|
}
|
|
}
|
|
|
|
void lxc_unload_script(struct lxc_States *state)
|
|
{
|
|
struct lxc_hooks *hooks, *h;
|
|
struct lxc_cbdata *cb;
|
|
struct lxc_userdata *ud, *u;
|
|
lua_State *L = state->state;
|
|
|
|
lua_pushstring(L, "xchat_unload");
|
|
lua_gettable(L, LUA_GLOBALSINDEX);
|
|
if (lua_type(L, -1) == LUA_TFUNCTION) {
|
|
if (lua_pcall(L, 0, 0, 0)) {
|
|
xchat_printf(ph, "Lua plugin: error while unloading script %s",
|
|
lua_tostring(L, -1));
|
|
lua_pop(L, 1);
|
|
}
|
|
}
|
|
|
|
if (state->gui)
|
|
xchat_plugingui_remove(ph, state->gui);
|
|
state->gui = NULL;
|
|
|
|
hooks = state->hooks;
|
|
while (hooks) {
|
|
h = hooks;
|
|
hooks = hooks->next;
|
|
|
|
cb = xchat_unhook(ph, h->hook);
|
|
if (cb) {
|
|
ud = cb->data;
|
|
while (ud) {
|
|
u = ud;
|
|
ud = ud->next;
|
|
free(u);
|
|
}
|
|
free(cb);
|
|
}
|
|
|
|
free(h);
|
|
}
|
|
lua_close(state->state);
|
|
}
|
|
|
|
|
|
static int lxc_cb_load(char *word[], char *word_eol[], void *userdata)
|
|
{
|
|
int len;
|
|
struct lxc_States *state;
|
|
lua_State *L;
|
|
const char *name, *desc, *vers;
|
|
const char *xdir = "";
|
|
char *buf;
|
|
char file[PATH_MAX+1];
|
|
struct stat *st;
|
|
|
|
if (word_eol[2][0] == 0)
|
|
return XCHAT_EAT_NONE;
|
|
|
|
buf = malloc(PATH_MAX + 1);
|
|
if (!buf) {
|
|
xchat_printf(ph, "malloc() failed: %s\n", strerror(errno));
|
|
return XCHAT_EAT_NONE;
|
|
}
|
|
|
|
st = malloc(sizeof(struct stat));
|
|
if (!st) {
|
|
xchat_printf(ph, "malloc() failed: %s\n", strerror(errno));
|
|
free(buf);
|
|
return XCHAT_EAT_NONE;
|
|
}
|
|
|
|
len = strlen(word[2]);
|
|
if (len > 4 && strcasecmp (".lua", word[2] + len - 4) == 0) {
|
|
#ifdef WIN32
|
|
if (strrchr(word[2], '\\') != NULL)
|
|
#else
|
|
if (strrchr(word[2], '/') != NULL)
|
|
#endif
|
|
strncpy(file, word[2], PATH_MAX);
|
|
else {
|
|
if (stat(word[2], st) == 0)
|
|
{
|
|
xdir = getcwd (buf, PATH_MAX);
|
|
snprintf (file, PATH_MAX, "%s/%s", xdir, word[2]);
|
|
}
|
|
else
|
|
{
|
|
xdir = xchat_get_info (ph, "xchatdirfs");
|
|
snprintf (file, PATH_MAX, "%s/addons/%s", xdir, word[2]);
|
|
}
|
|
}
|
|
|
|
if (lxc_load_file((const char *)file) == 0) {
|
|
free(st);
|
|
free(buf);
|
|
return XCHAT_EAT_ALL;
|
|
}
|
|
|
|
state = lxc_states;
|
|
while (state) {
|
|
if (state->next == NULL) {
|
|
L = state->state;
|
|
|
|
lua_pushstring(L, "xchat_register");
|
|
lua_gettable(L, LUA_GLOBALSINDEX);
|
|
if (lua_pcall(L, 0, 3, 0)) {
|
|
xchat_printf(ph, "Lua plugin: error registering script %s",
|
|
lua_tostring(L, -1));
|
|
lua_pop(L, 1);
|
|
free(st);
|
|
free(buf);
|
|
return XCHAT_EAT_ALL;
|
|
}
|
|
|
|
name = lua_tostring(L, -3);
|
|
desc = lua_tostring(L, -2);
|
|
vers = lua_tostring(L, -1);
|
|
lua_pop(L, 4); /* func + 3 ret value */
|
|
state->gui = xchat_plugingui_add(ph, state->file,
|
|
name, desc, vers, NULL
|
|
);
|
|
|
|
lua_pushstring(L, "xchat_init");
|
|
lua_gettable(L, LUA_GLOBALSINDEX);
|
|
if (lua_type(L, -1) != LUA_TFUNCTION)
|
|
lua_pop(L, 1);
|
|
else {
|
|
if (lua_pcall(L, 0, 0, 0)) {
|
|
xchat_printf(ph,
|
|
"Lua plugin: error calling xchat_init() %s",
|
|
lua_tostring(L, -1));
|
|
lua_pop(L, 1);
|
|
}
|
|
}
|
|
free(st);
|
|
free(buf);
|
|
return XCHAT_EAT_ALL;
|
|
}
|
|
state = state->next;
|
|
}
|
|
}
|
|
free(st);
|
|
free(buf);
|
|
return XCHAT_EAT_NONE;
|
|
}
|
|
|
|
static int lxc_cb_unload(char *word[], char *word_eol[], void *userdata)
|
|
{
|
|
int len;
|
|
struct lxc_States *state;
|
|
struct lxc_States *prev = NULL;
|
|
char *file;
|
|
|
|
if (word_eol[2][0] == 0)
|
|
return XCHAT_EAT_NONE;
|
|
|
|
len = strlen(word[2]);
|
|
if (len > 4 && strcasecmp(".lua", word[2] + len - 4) == 0) {
|
|
state = lxc_states;
|
|
while (state) {
|
|
/*
|
|
* state->file is the full or relative path, always with a '/' inside,
|
|
* even if loaded via '/LOAD script.lua'. So strrchr() will never
|
|
* be NULL.
|
|
* ... we just inspect the script name w/o path to see if it's the
|
|
* right one to unload
|
|
*/
|
|
file = strrchr(state->file, '/') + 1;
|
|
if ((strcmp(state->file, word[2]) == 0)
|
|
|| (strcasecmp(file, word[2]) == 0)) {
|
|
lxc_unload_script(state);
|
|
if (prev)
|
|
prev->next = state->next;
|
|
else
|
|
lxc_states = state->next;
|
|
xchat_printf(ph, "Lua script %s unloaded", file);
|
|
free(state);
|
|
return XCHAT_EAT_ALL;
|
|
}
|
|
prev = state;
|
|
state = state->next;
|
|
}
|
|
}
|
|
return XCHAT_EAT_NONE;
|
|
}
|
|
|
|
static int lxc_cb_lua(char *word[], char *word_eol[], void *userdata)
|
|
{
|
|
lua_State *L = lxc_new_state();
|
|
if (word[2][0] == '\0') {
|
|
xchat_printf(ph, "LUA: Usage: /LUA LUA_CODE... execute LUA_CODE");
|
|
return XCHAT_EAT_ALL;
|
|
}
|
|
if (luaL_loadbuffer(L, word_eol[2], strlen(word_eol[2]), "/LUA")) {
|
|
xchat_printf(ph, "LUA: error loading line %s", lua_tostring(L, -1));
|
|
lua_pop(L, 1);
|
|
}
|
|
|
|
#define LXC_HOOK_DISABLE "xchat.hook_command = nil\n" \
|
|
"xchat.hook_server = nil\n" \
|
|
"xchat.hook_print = nil\n" \
|
|
"xchat.hook_timer = nil\n"
|
|
|
|
#if defined(LUA_VERSION_NUM) && (LUA_VERSION_NUM >= 501)
|
|
(void)luaL_dostring(L, LXC_HOOK_DISABLE);
|
|
#else
|
|
lua_dostring(L, LXC_HOOK_DISABLE);
|
|
#endif
|
|
|
|
if (lua_pcall(L, 0, 0, 0)) {
|
|
xchat_printf(ph, "LUA: error executing line %s", lua_tostring(L, -1));
|
|
lua_pop(L, 1);
|
|
}
|
|
|
|
lua_close(L);
|
|
return XCHAT_EAT_ALL;
|
|
}
|
|
|
|
int xchat_plugin_init(xchat_plugin *plugin_handle,
|
|
char **plugin_name,
|
|
char **plugin_desc,
|
|
char **plugin_version,
|
|
char *arg)
|
|
{
|
|
struct lxc_States *state;
|
|
lua_State *L;
|
|
const char *xdir;
|
|
const char *name, *desc, *vers;
|
|
char *xsubdir;
|
|
/* we need to save this for use with any xchat_* functions */
|
|
ph = plugin_handle;
|
|
|
|
/* tell xchat our info */
|
|
*plugin_name = LXC_NAME;
|
|
*plugin_desc = LXC_DESC;
|
|
*plugin_version = LXC_VERSION;
|
|
|
|
xchat_hook_command(ph, "LOAD", XCHAT_PRI_NORM, lxc_cb_load, NULL, NULL);
|
|
xchat_hook_command(ph, "UNLOAD", XCHAT_PRI_NORM, lxc_cb_unload, NULL, NULL);
|
|
xchat_hook_command(ph, "LUA", XCHAT_PRI_NORM, lxc_cb_lua, "Usage: LUA <code>, executes <code> in a new lua state", NULL);
|
|
|
|
xdir = xchat_get_info (ph, "xchatdirfs");
|
|
xsubdir = g_build_filename (xdir, "addons", NULL);
|
|
lxc_autoload_from_path (xsubdir);
|
|
g_free (xsubdir);
|
|
|
|
/* put this here, otherwise it's only displayed when a script is autoloaded upon start */
|
|
xchat_printf(ph, "Lua interface loaded");
|
|
|
|
if (!lxc_states) /* no scripts loaded */
|
|
return 1;
|
|
|
|
state = lxc_states;
|
|
while (state) {
|
|
L = state->state;
|
|
lua_pushstring(L, "xchat_register");
|
|
lua_gettable(L, LUA_GLOBALSINDEX);
|
|
if (lua_pcall(L, 0, 3, 0)) {
|
|
xchat_printf(ph, "Lua plugin: error registering script %s",
|
|
lua_tostring(L, -1));
|
|
lua_pop(L, 1);
|
|
state = state->next;
|
|
continue;
|
|
}
|
|
|
|
name = lua_tostring(L, -3);
|
|
desc = lua_tostring(L, -2);
|
|
vers = lua_tostring(L, -1);
|
|
lua_pop(L, 4); /* func + 3 ret value */
|
|
state->gui = xchat_plugingui_add(ph, state->file, name, desc, vers, NULL);
|
|
|
|
lua_pushstring(L, "xchat_init");
|
|
lua_gettable(L, LUA_GLOBALSINDEX);
|
|
if (lua_type(L, -1) != LUA_TFUNCTION)
|
|
lua_pop(L, 1);
|
|
else {
|
|
if (lua_pcall(L, 0, 0, 0)) {
|
|
xchat_printf(ph, "Lua plugin: error calling xchat_init() %s",
|
|
lua_tostring(L, -1));
|
|
lua_pop(L, 1);
|
|
}
|
|
}
|
|
state = state->next;
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
int xchat_plugin_deinit(xchat_plugin *plug_handle)
|
|
{
|
|
struct lxc_States *state, *st;
|
|
|
|
state = lxc_states;
|
|
while (state) {
|
|
lxc_unload_script(state);
|
|
xchat_printf(ph, "Lua script %s unloaded", state->file);
|
|
st = state;
|
|
state = state->next;
|
|
free(st);
|
|
}
|
|
xchat_printf(plug_handle, "Lua interface unloaded");
|
|
return 1;
|
|
}
|
|
|
|
/*
|
|
* lua: func_name(word, word_eol, data)
|
|
* desc: your previously hooked callback function for hook_command() and
|
|
* hook_server(), you must return one of the xchat.EAT_* constants
|
|
* ret: none
|
|
* args:
|
|
* * word (table): the incoming line split into words (max 32)
|
|
* * word_eol (table):
|
|
* for both see
|
|
* http://xchat.org/docs/plugin20.html#word
|
|
* * data (table): the data table you passed to the hook_command() /
|
|
* hook_server() as 5th arg
|
|
*/
|
|
static int lxc_run_hook(char *word[], char *word_eol[], void *data)
|
|
{
|
|
struct lxc_cbdata *cb = data;
|
|
lua_State *L = cb->state;
|
|
struct lxc_userdata *ud = cb->data;
|
|
struct lxc_userdata *u;
|
|
int i;
|
|
lua_pushstring(L, cb->func);
|
|
lua_gettable(L, LUA_GLOBALSINDEX);
|
|
|
|
strcpy(lxc_event_name, word[0]);
|
|
lua_newtable(L);
|
|
for (i=1; i<=31 && word[i][0]; i++) {
|
|
lua_pushnumber(L, i);
|
|
lua_pushstring(L, word[i]);
|
|
lua_settable(L, -3);
|
|
}
|
|
|
|
lua_newtable(L);
|
|
for (i=1; i<=31 && word_eol[i][0]; i++) {
|
|
lua_pushnumber(L, i);
|
|
lua_pushstring(L, word_eol[i]);
|
|
lua_settable(L, -3);
|
|
}
|
|
|
|
lua_newtable(L);
|
|
u = ud;
|
|
while (u) {
|
|
lua_pushnumber(L, u->idx);
|
|
switch (u->type) {
|
|
case LUA_TSTRING:
|
|
lua_pushstring(L, u->string);
|
|
break;
|
|
case LUA_TNUMBER:
|
|
lua_pushnumber(L, u->num);
|
|
break;
|
|
case LUA_TBOOLEAN:
|
|
lua_pushboolean(L, (((int)u->num == 0) ? 0 : 1));
|
|
break;
|
|
default: /* LUA_TNIL or others */
|
|
lua_pushnil(L);
|
|
break;
|
|
}
|
|
lua_settable(L, -3);
|
|
u = u->next;
|
|
}
|
|
|
|
if (lua_pcall(L, 3, 1, 0)) {
|
|
xchat_printf(ph, "failed to call callback for '%s': %s",
|
|
word[1], lua_tostring(L, -1)
|
|
);
|
|
lua_pop(L, 1);
|
|
return XCHAT_EAT_NONE;
|
|
}
|
|
|
|
if (lua_type(L, -1) != LUA_TNUMBER) {
|
|
xchat_printf(ph, "callback for '%s' did not return number...", word[1]);
|
|
return XCHAT_EAT_NONE;
|
|
}
|
|
|
|
i = (int)lua_tonumber(L, -1);
|
|
lua_pop(L, 1);
|
|
return i;
|
|
}
|
|
|
|
static int lxc_get_userdata(int pos, struct lxc_cbdata *cb)
|
|
{
|
|
struct lxc_userdata *ud, *u;
|
|
lua_State *L = cb->state;
|
|
int i, t;
|
|
|
|
t = lua_type(L, pos);
|
|
if (t == LUA_TNIL)
|
|
return 1;
|
|
if (t != LUA_TTABLE)
|
|
return 0;
|
|
|
|
i = 1;
|
|
while (1) {
|
|
lua_pushnumber(L, i);
|
|
lua_gettable(L, -2);
|
|
|
|
t = lua_type(L, -1);
|
|
if (t == LUA_TNIL) {
|
|
lua_pop(L, 1);
|
|
break;
|
|
}
|
|
|
|
ud = malloc(sizeof(struct lxc_userdata));
|
|
if (!ud) {
|
|
xchat_printf(ph, "lxc_get_userdata(): failed to malloc: %s",
|
|
strerror(errno));
|
|
if (cb->data != NULL) {
|
|
ud = cb->data;
|
|
while (ud) {
|
|
u = ud;
|
|
ud = ud->next;
|
|
free(u);
|
|
}
|
|
}
|
|
/* free(cb); NO! */
|
|
lua_pushnil(L);
|
|
return 0;
|
|
}
|
|
ud->idx = i;
|
|
ud->next = NULL;
|
|
switch (t) {
|
|
case LUA_TSTRING:
|
|
ud->string = lua_tostring(L, -1);
|
|
ud->type = LUA_TSTRING;
|
|
break;
|
|
case LUA_TNUMBER:
|
|
ud->num = lua_tonumber(L, -1);
|
|
ud->type = LUA_TNUMBER;
|
|
break;
|
|
case LUA_TBOOLEAN:
|
|
ud->num = (double)lua_toboolean(L, -1);
|
|
ud->type = LUA_TBOOLEAN;
|
|
break;
|
|
default:
|
|
ud->type = LUA_TNIL;
|
|
break;
|
|
}
|
|
lua_pop(L, 1);
|
|
|
|
if (cb->data == NULL)
|
|
cb->data = ud;
|
|
else {
|
|
u = cb->data;
|
|
while (u->next)
|
|
u = u->next;
|
|
u->next = ud;
|
|
}
|
|
i++;
|
|
} /* END while (1) */
|
|
return 1;
|
|
}
|
|
|
|
/*
|
|
* lua: xchat.hook_command(name, func_name, prio, help_str, data)
|
|
* desc: Adds a new /command. This allows your program to handle commands
|
|
* entered at the input box. To capture text without a "/" at the start
|
|
* (non-commands), you may hook a special name of "". i.e
|
|
* xchat.hook_command( "", ...)
|
|
* Starting from version 2.6.8, commands hooked that begin with a
|
|
* period ('.') will be hidden in /HELP and /HELP -l.
|
|
* ret: true... or false if something went wrong while registering hook
|
|
* args:
|
|
* * name (string): the name of the new command
|
|
* * func_name (string): the lua function to be called when command is
|
|
* entered
|
|
* * prio (number): use one of the xchat.PRIO_*
|
|
* * help_str (string): help for the new command... use nil for no help
|
|
* * data (table): table with strings, numbers and booleans, which will
|
|
* be passed to func_name as last argument.
|
|
*/
|
|
static int lxc_hook_command(lua_State *L)
|
|
{
|
|
xchat_hook *hook;
|
|
const char *help, *command, *func;
|
|
double prio;
|
|
struct lxc_hooks *hooks, *h;
|
|
struct lxc_States *st;
|
|
struct lxc_cbdata *cb;
|
|
|
|
|
|
if (lua_gettop(L) < 5) /* expand to five args if necessary */
|
|
lua_settop(L, 5);
|
|
|
|
cb = malloc(sizeof(struct lxc_cbdata));
|
|
if (!cb) {
|
|
xchat_printf(ph, "lxc_hook_command(): failed to malloc: %s",
|
|
strerror(errno));
|
|
lua_pushboolean(L, 0);
|
|
return 1;
|
|
}
|
|
|
|
cb->state = L;
|
|
cb->data = NULL;
|
|
|
|
command = luaL_checkstring(L, 1);
|
|
func = luaL_checkstring(L, 2);
|
|
cb->func = func;
|
|
cb->hook = NULL;
|
|
|
|
if (lua_type(L, 3) == LUA_TNIL)
|
|
prio = XCHAT_PRI_NORM;
|
|
else
|
|
prio = luaL_checknumber(L, 3);
|
|
|
|
if (lua_type(L, 4) == LUA_TSTRING) {
|
|
help = luaL_checkstring(L, 4);
|
|
if (strlen(help) == 0)
|
|
help = NULL;
|
|
}
|
|
else
|
|
help = NULL;
|
|
|
|
if (lxc_get_userdata(5, cb) == 0)
|
|
lua_pushboolean(L, 0);
|
|
else {
|
|
h = malloc(sizeof(struct lxc_hooks));
|
|
if (!h) {
|
|
xchat_printf(ph, "lxc_hook_command(): failed to malloc: %s",
|
|
strerror(errno));
|
|
lua_pushboolean(L, 0);
|
|
return 1;
|
|
}
|
|
hook = xchat_hook_command(ph, command, prio, lxc_run_hook, help, cb);
|
|
h->hook = hook;
|
|
h->name = command;
|
|
h->next = NULL;
|
|
st = lxc_states;
|
|
while (st) {
|
|
if (st->state == L) {
|
|
if (!st->hooks)
|
|
st->hooks = h;
|
|
else {
|
|
hooks = st->hooks;
|
|
while (hooks->next)
|
|
hooks = hooks->next;
|
|
hooks->next = h;
|
|
}
|
|
break;
|
|
}
|
|
st = st->next;
|
|
}
|
|
lua_pushboolean(L, 1);
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
/*
|
|
* lua: func_name(word, data)
|
|
* desc: your previously hooked callback function for hook_print(),
|
|
* you must return one of the xchat.EAT_* constants
|
|
* ret: none
|
|
* args:
|
|
* * word (table): the incoming line split into words (max 32)
|
|
* (see http://xchat.org/docs/plugin20.html#word)
|
|
* * data (table): the data table you passed to the hook_print() /
|
|
* as 4th arg
|
|
*/
|
|
static int lxc_run_print(char *word[], void *data)
|
|
{
|
|
struct lxc_cbdata *cb = data;
|
|
lua_State *L = cb->state;
|
|
int i;
|
|
|
|
lua_pushstring(L, cb->func);
|
|
lua_gettable(L, LUA_GLOBALSINDEX);
|
|
|
|
strcpy(lxc_event_name, word[0]);
|
|
lua_newtable(L);
|
|
for (i=1; i<=31 && word[i][0]; i++) {
|
|
lua_pushnumber(L, i);
|
|
lua_pushstring(L, word[i]);
|
|
lua_settable(L, -3);
|
|
}
|
|
|
|
if (lua_pcall(L, 1, 1, 0)) {
|
|
xchat_printf(ph, "failed to call callback for '%s': %s",
|
|
word[1], lua_tostring(L, -1));
|
|
lua_pop(L, 1);
|
|
return 0;
|
|
}
|
|
|
|
if (lua_type(L, -1) != LUA_TNUMBER) {
|
|
xchat_printf(ph, "callback for '%s' didn't return number...", word[1]);
|
|
return XCHAT_EAT_NONE;
|
|
}
|
|
i = (int)lua_tonumber(L, -1);
|
|
lua_pop(L, 1);
|
|
return i;
|
|
}
|
|
|
|
/*
|
|
* lua: xchat.hook_print(name, func_name, prio, data)
|
|
* desc: Registers a function to trap any print events. The event names may
|
|
* be any available in the "Advanced > Text Events" window. There are
|
|
* also some extra "special" events you may hook using this function,
|
|
* see: http://xchat.org/docs/plugin20.html#xchat_hook_print
|
|
* ret: true... or false if something went wrong while registering hook
|
|
* args:
|
|
* * name (string): the name of the new command
|
|
* * prio (number): use one of the xchat.PRIO_*
|
|
* * func_name (string): the lua function to be called when command is
|
|
* entered
|
|
* * data (table): table with strings, numbers and booleans, which will
|
|
* be passed to func_name as last argument.
|
|
*/
|
|
static int lxc_hook_print(lua_State *L)
|
|
{
|
|
xchat_hook *hook;
|
|
struct lxc_hooks *hooks, *h;
|
|
struct lxc_States *st;
|
|
struct lxc_cbdata *cb = malloc(sizeof(struct lxc_cbdata));
|
|
const char *name, *func;
|
|
double prio;
|
|
|
|
if (!cb) {
|
|
luaL_error(L, "lxc_hook_print(): failed to malloc: %s", strerror(errno));
|
|
return 0;
|
|
}
|
|
|
|
if (lua_gettop(L) < 4) /* expand to 4 args if necessary */
|
|
lua_settop(L, 4);
|
|
|
|
name = luaL_checkstring(L, 1);
|
|
func = luaL_checkstring(L, 2);
|
|
if (lua_type(L, 3) == LUA_TNIL)
|
|
prio = XCHAT_PRI_NORM;
|
|
else
|
|
prio = luaL_checknumber(L, 3);
|
|
|
|
cb->state = L;
|
|
cb->func = func;
|
|
cb->data = NULL;
|
|
cb->hook = NULL;
|
|
|
|
if (lxc_get_userdata(4, cb) == 0)
|
|
lua_pushboolean(L, 0);
|
|
else {
|
|
h = malloc(sizeof(struct lxc_hooks));
|
|
if (!h) {
|
|
xchat_printf(ph, "lxc_hook_print(): failed to malloc: %s",
|
|
strerror(errno));
|
|
lua_pushboolean(L, 0);
|
|
return 1;
|
|
}
|
|
hook = xchat_hook_print(ph, name, prio, lxc_run_print, cb);
|
|
h->hook = hook;
|
|
h->name = name;
|
|
h->next = NULL;
|
|
st = lxc_states;
|
|
while (st) {
|
|
if (st->state == L) {
|
|
if (!st->hooks)
|
|
st->hooks = h;
|
|
else {
|
|
hooks = st->hooks;
|
|
while (hooks->next)
|
|
hooks = hooks->next;
|
|
hooks->next = h;
|
|
}
|
|
break;
|
|
}
|
|
st = st->next;
|
|
}
|
|
lua_pushboolean(L, 1);
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
/*
|
|
* lua: xchat.hook_server(name, func_name, prio, data)
|
|
* desc: Registers a function to be called when a certain server event
|
|
* occurs. You can use this to trap PRIVMSG, NOTICE, PART, a server
|
|
* numeric etc... If you want to hook every line that comes from the
|
|
* IRC server, you may use the special name of "RAW LINE".
|
|
* ret: true... or false if something went wrong while registering
|
|
* args:
|
|
* * name (string): the event name / numeric (yes, also as a string)
|
|
* * prio (number): one of the xchat.PRIO_* constants
|
|
* * func_name (string): the function to be called, when the event
|
|
* happens
|
|
* * data (table)... see xchat.hook_command()
|
|
*/
|
|
static int lxc_hook_server(lua_State *L)
|
|
{
|
|
xchat_hook *hook;
|
|
struct lxc_hooks *hooks, *h;
|
|
struct lxc_States *st;
|
|
const char *name, *func;
|
|
double prio;
|
|
|
|
struct lxc_cbdata *cb = malloc(sizeof(struct lxc_cbdata));
|
|
if (!cb) {
|
|
xchat_printf(ph, "lxc_hook_server(): failed to malloc: %s",
|
|
strerror(errno));
|
|
lua_pushnil(L);
|
|
return 1;
|
|
}
|
|
|
|
if (lua_gettop(L) < 4) /* expand to 4 args if necessary */
|
|
lua_settop(L, 4);
|
|
|
|
name = luaL_checkstring(L, 1);
|
|
func = luaL_checkstring(L, 2);
|
|
if (lua_type(L, 3) == LUA_TNIL)
|
|
prio = XCHAT_PRI_NORM;
|
|
else
|
|
prio = luaL_checknumber(L, 3);
|
|
|
|
cb->state = L;
|
|
cb->func = func;
|
|
cb->data = NULL;
|
|
cb->hook = NULL;
|
|
|
|
if (lxc_get_userdata(4, cb) == 0)
|
|
lua_pushboolean(L, 0);
|
|
else {
|
|
h = malloc(sizeof(struct lxc_hooks));
|
|
if (!h) {
|
|
xchat_printf(ph, "lxc_hook_server(): failed to malloc: %s",
|
|
strerror(errno));
|
|
lua_pushboolean(L, 0);
|
|
return 1;
|
|
}
|
|
hook = xchat_hook_server(ph, name, prio, lxc_run_hook, cb);
|
|
h->hook = hook;
|
|
h->name = name;
|
|
h->next = NULL;
|
|
st = lxc_states;
|
|
while (st) {
|
|
if (st->state == L) {
|
|
if (!st->hooks)
|
|
st->hooks = h;
|
|
else {
|
|
hooks = st->hooks;
|
|
while (hooks->next)
|
|
hooks = hooks->next;
|
|
hooks->next = h;
|
|
}
|
|
break;
|
|
}
|
|
st = st->next;
|
|
}
|
|
lua_pushboolean(L, 1);
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
/*
|
|
* lua: xchat.hook_timer(timeout, func_name, data)
|
|
* desc: Registers a function to be called every "timeout" milliseconds.
|
|
* ret: true (or false on error while registering)
|
|
* args:
|
|
* * timeout (number): Timeout in milliseconds (1000 is 1 second).
|
|
* * func_name (string): Callback function. This will be called
|
|
* every "timeout" milliseconds.
|
|
* * data (table): see xchat.hook_command()
|
|
*/
|
|
|
|
static unsigned long long lxc_timer_count = 0;
|
|
|
|
static int lxc_hook_timer(lua_State *L)
|
|
{
|
|
xchat_hook *hook;
|
|
struct lxc_hooks *hooks, *h;
|
|
struct lxc_States *st;
|
|
double timeout;
|
|
const char *func;
|
|
char name[32];
|
|
|
|
struct lxc_cbdata *cb = malloc(sizeof(struct lxc_cbdata));
|
|
if (!cb) {
|
|
luaL_error(L, "lxc_hook_timer(): failed to malloc: %s", strerror(errno));
|
|
lua_pushnil(L);
|
|
return 1;
|
|
}
|
|
|
|
if (lua_gettop(L) < 3) /* expand to 3 args if necessary */
|
|
lua_settop(L, 3);
|
|
|
|
timeout = luaL_checknumber(L, 1);
|
|
func = luaL_checkstring(L, 2);
|
|
|
|
cb->state = L;
|
|
cb->func = func;
|
|
cb->data = NULL;
|
|
|
|
if (lxc_get_userdata(3, cb) == 0)
|
|
lua_pushnil(L);
|
|
else {
|
|
h = malloc(sizeof(struct lxc_hooks));
|
|
if (!h) {
|
|
luaL_error(L, "lxc_hook_timer(): failed to malloc: %s",
|
|
strerror(errno));
|
|
return 0;
|
|
}
|
|
hook = xchat_hook_timer(ph, timeout, lxc_run_timer, cb);
|
|
cb->hook = hook;
|
|
h->hook = hook;
|
|
h->next = NULL;
|
|
snprintf(name, 31, "timer%llu", lxc_timer_count++);
|
|
h->name = name;
|
|
lua_pushstring(L, name);
|
|
|
|
st = lxc_states;
|
|
while (st) {
|
|
if (st->state == L) {
|
|
if (!st->hooks)
|
|
st->hooks = h;
|
|
else {
|
|
hooks = st->hooks;
|
|
while (hooks->next)
|
|
hooks = hooks->next;
|
|
hooks->next = h;
|
|
}
|
|
break;
|
|
}
|
|
st = st->next;
|
|
}
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
static void lxc_unhook_timer(lua_State *L, xchat_hook *hook)
|
|
{
|
|
struct lxc_States *state;
|
|
struct lxc_hooks *hooks, *h, *prev_hook;
|
|
struct lxc_cbdata *cb;
|
|
struct lxc_userdata *ud, *u;
|
|
|
|
prev_hook = NULL;
|
|
state = lxc_states;
|
|
while (state) {
|
|
if (state->state == L) {
|
|
hooks = state->hooks;
|
|
while (hooks) {
|
|
if (hooks->hook == hook) {
|
|
h = hooks;
|
|
if (prev_hook)
|
|
prev_hook->next = hooks->next;
|
|
else
|
|
state->hooks = hooks->next;
|
|
|
|
cb = xchat_unhook(ph, h->hook);
|
|
if (cb) {
|
|
ud = cb->data;
|
|
while (ud) {
|
|
u = ud;
|
|
ud = ud->next;
|
|
free(u);
|
|
}
|
|
free(cb);
|
|
}
|
|
|
|
free(h);
|
|
return;
|
|
}
|
|
prev_hook = hooks;
|
|
hooks = hooks->next;
|
|
}
|
|
break;
|
|
}
|
|
state = state->next;
|
|
}
|
|
}
|
|
|
|
/*
|
|
* lua: func_name(data)
|
|
* desc: the callback function for the registered timer hook, return
|
|
* true to keep this timer going, false to stop it
|
|
* ret: none
|
|
* args:
|
|
* * data (table): the table you gave the hook_timer() as last
|
|
* argument
|
|
*/
|
|
static int lxc_run_timer(void *data)
|
|
{
|
|
int ret;
|
|
struct lxc_cbdata *cb = data;
|
|
xchat_hook *hook = cb->hook;
|
|
lua_State *L = cb->state;
|
|
|
|
lua_pushstring(L, cb->func);
|
|
lua_gettable(L, LUA_GLOBALSINDEX);
|
|
|
|
if (lua_pcall(L, 0, 1, 0)) {
|
|
xchat_printf(ph, "failed to call timer callback for '%s': %s",
|
|
cb->func, lua_tostring(L, -1));
|
|
lua_pop(L, 1);
|
|
lxc_unhook_timer(L, hook);
|
|
return 0;
|
|
}
|
|
|
|
if (lua_type(L, -1) != LUA_TBOOLEAN) {
|
|
xchat_printf(ph,
|
|
"timer callback for '%s' didn't return a boolean", cb->func);
|
|
lua_pop(L, 1);
|
|
lxc_unhook_timer(L, hook);
|
|
return 0;
|
|
}
|
|
|
|
ret = (lua_toboolean(L, -1) == 0) ? 0 : 1;
|
|
lua_pop(L, 1);
|
|
|
|
if (ret == 0)
|
|
lxc_unhook_timer(L, hook);
|
|
|
|
return ret;
|
|
}
|
|
|
|
/*
|
|
* lua: xchat.unhook(name)
|
|
* desc: unhooks a previously hooked hook
|
|
* ret: true if the hook existed, else false..
|
|
* args:
|
|
* * name (string): name of a registered hook (e.g. with
|
|
* xchat.hook_command("whois", ... ) you would unhook "whois"
|
|
* ... see timer warnings... there's currently just one "timer"
|
|
* to unhook
|
|
*/
|
|
static int lxc_unhook(lua_State *L)
|
|
{
|
|
struct lxc_States *state;
|
|
struct lxc_hooks *hooks, *h, *prev_hook;
|
|
struct lxc_cbdata *cb;
|
|
struct lxc_userdata *ud, *u;
|
|
int done = 0;
|
|
const char *name = luaL_checkstring(L, 1);
|
|
|
|
prev_hook = NULL;
|
|
state = lxc_states;
|
|
while (state) {
|
|
if (state->state == L) {
|
|
hooks = state->hooks;
|
|
while (hooks) {
|
|
if (strcasecmp(hooks->name, name) == 0) {
|
|
h = hooks;
|
|
if (prev_hook)
|
|
prev_hook->next = hooks->next;
|
|
else
|
|
state->hooks = hooks->next;
|
|
|
|
cb = xchat_unhook(ph, h->hook);
|
|
if (cb) {
|
|
ud = cb->data;
|
|
while (ud) {
|
|
u = ud;
|
|
ud = ud->next;
|
|
free(u);
|
|
}
|
|
free(cb);
|
|
}
|
|
|
|
free(h);
|
|
done = 1;
|
|
break;
|
|
}
|
|
prev_hook = hooks;
|
|
hooks = hooks->next;
|
|
}
|
|
break;
|
|
}
|
|
state = state->next;
|
|
}
|
|
lua_pushboolean(L, done);
|
|
return 1;
|
|
}
|
|
|
|
static int lxc_event(lua_State *L)
|
|
{
|
|
lua_pushstring(L, lxc_event_name);
|
|
return 1;
|
|
}
|
|
|
|
/*
|
|
* lua: xchat.command(command)
|
|
* desc: executes a command as if it were typed in xchat's input box.
|
|
* ret: none
|
|
* args:
|
|
* * command (string): command to execute, without the forward slash "/".
|
|
*/
|
|
static int lxc_command(lua_State *L)
|
|
{
|
|
const char *command = luaL_checkstring(L, 1);
|
|
xchat_command(ph, command);
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* lua: xchat.print(text)
|
|
* desc: Prints some text to the current tab/window.
|
|
* ret: none
|
|
* args:
|
|
* * text (string): the text to print
|
|
*/
|
|
static int lxc_print(lua_State *L)
|
|
{
|
|
const char *txt = luaL_checkstring(L, 1);
|
|
// FIXME? const char *txt = lua_tostring(L, 1);
|
|
xchat_print(ph, txt);
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* lua: xchat.emit_print(event, text, [text2, ...])
|
|
* desc: Generates a print event. This can be any event found in the
|
|
* Preferences > Advanced > Text Events window. The vararg parameter
|
|
* list MUST be no longer than four (4) parameters.
|
|
* Special care should be taken when calling this function inside a
|
|
* print callback (from xchat.hook_print()), as not to cause endless
|
|
* recursion.
|
|
* ret: true on success, false on error
|
|
* args:
|
|
* * event (string): the event name from the references > Advanced >
|
|
* Text Events window
|
|
* * text (string)
|
|
* text2 (string)
|
|
* ... (string(s)):
|
|
* parameters for the given event
|
|
*/
|
|
static int lxc_emit_print(lua_State *L)
|
|
{
|
|
|
|
int n = lua_gettop(L);
|
|
const char *text[5];
|
|
const char *event;
|
|
int i = 2;
|
|
|
|
if (n > 6)
|
|
luaL_error(L, "too many arguments to xchat.emit_print()");
|
|
|
|
event = luaL_checkstring(L, 1);
|
|
while (i <= n) {
|
|
text[i-2] = luaL_checkstring(L, i);
|
|
i++;
|
|
}
|
|
switch (n-1) {
|
|
case 0:
|
|
i = xchat_emit_print(ph, event, NULL);
|
|
break;
|
|
case 1:
|
|
i = xchat_emit_print(ph, event, text[0], NULL);
|
|
break;
|
|
case 2:
|
|
i = xchat_emit_print(ph, event, text[0], text[1], NULL);
|
|
break;
|
|
case 3:
|
|
i = xchat_emit_print(ph, event, text[0], text[1], text[2], NULL);
|
|
break;
|
|
case 4:
|
|
i = xchat_emit_print(ph, event, text[0], text[1], text[2], text[3], NULL);
|
|
break;
|
|
}
|
|
lua_pushboolean(L, (i == 0) ? 0 : 1);
|
|
return 1;
|
|
}
|
|
|
|
/*
|
|
* lua: xchat.send_modes(targets, sign, mode [, modes_per_line])
|
|
* desc: Sends a number of channel mode changes to the current channel.
|
|
* For example, you can Op a whole group of people in one go. It may
|
|
* send multiple MODE lines if the request doesn't fit on one. Pass 0
|
|
* for modes_per_line to use the current server's maximum possible.
|
|
* This function should only be called while in a channel context.
|
|
* ret: none
|
|
* args:
|
|
* * targets (table): list of names
|
|
* * sign (string): mode sign, i.e. "+" or "-", only the first char of
|
|
* this is used (currently unchecked if it's really "+" or "-")
|
|
* * mode (string): mode char, i.e. "o" for opping, only the first
|
|
* char of this is used (currently unchecked, what char)
|
|
* * modes_per_line (number): [optional] number of modes per line
|
|
*/
|
|
static int lxc_send_modes(lua_State *L)
|
|
{
|
|
int i = 1;
|
|
const char *name, *mode, *sign;
|
|
const char *targets[4096];
|
|
int num = 0; /* modes per line */
|
|
|
|
if (!lua_istable(L, 1)) {
|
|
luaL_error(L,
|
|
"xchat.send_modes(): first argument is not a table: %s",
|
|
lua_typename(L, lua_type(L, 1)));
|
|
return 0;
|
|
}
|
|
|
|
while (1) {
|
|
lua_pushnumber(L, i); /* push index on stack */
|
|
lua_gettable(L, 1); /* and get the element @ index */
|
|
if (lua_isnil(L, -1)) { /* end of table */
|
|
lua_pop(L, 1);
|
|
break;
|
|
}
|
|
|
|
if (lua_type(L, -1) != LUA_TSTRING) { /* oops, something wrong */
|
|
luaL_error(L,
|
|
"lua: xchat.send_modes(): table element #%d not a string: %s",
|
|
i, lua_typename(L, lua_type(L, -1)));
|
|
lua_pop(L, 1);
|
|
return 0;
|
|
}
|
|
|
|
name = lua_tostring(L, -1);
|
|
if (name == NULL) { /* this should not happen, but... */
|
|
lua_pop(L, 1);
|
|
break;
|
|
}
|
|
|
|
targets[i-1] = name;
|
|
lua_pop(L, 1); /* take index from stack */
|
|
++i;
|
|
}
|
|
|
|
sign = luaL_checkstring(L, 2);
|
|
if (sign[0] == '\0' || sign[1] != '\0') {
|
|
luaL_error(L, "argument #2 (mode sign) does not have length 1");
|
|
return 0;
|
|
}
|
|
if ((sign[0] != '+') && (sign[0] != '-')) {
|
|
luaL_error(L, "argument #2 (mode sign) is not '+' or '-'");
|
|
return 0;
|
|
}
|
|
|
|
mode = luaL_checkstring(L, 3);
|
|
if (mode[0] == '\0' || mode[1] != '\0') {
|
|
luaL_error(L, "argument #3 (mode char) does not have length 1");
|
|
return 0;
|
|
}
|
|
if (!isalpha((int)mode[0]) || !isascii((int)mode[0])) {
|
|
luaL_error(L, "argument #3 is not a valid mode character");
|
|
return 0;
|
|
}
|
|
|
|
if (lua_gettop(L) == 4)
|
|
num = luaL_checknumber(L, 4);
|
|
|
|
xchat_send_modes(ph, targets, i-1, num, sign[0], mode[0]);
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* lua: xchat.find_context(srv, chan)
|
|
* desc: Finds a context based on a channel and servername. If servname is nil,
|
|
* it finds any channel (or query) by the given name. If channel is nil,
|
|
* it finds the front-most tab/window of the given servname. If nil is
|
|
* given for both arguments, the currently focused tab/window will be
|
|
* returned.
|
|
* Changed in 2.6.1. If servname is nil, it finds the channel (or query)
|
|
* by the given name in the same server group as the current context.
|
|
* If that doesn't exists then find any by the given name.
|
|
* ret: context number (DON'T modify)
|
|
* args:
|
|
* * srv (string or nil): server name
|
|
* * chan (string or nil): channel / query name
|
|
*/
|
|
static int lxc_find_context(lua_State *L)
|
|
{
|
|
const char *srv, *chan;
|
|
long ctx;
|
|
xchat_context *ptr;
|
|
|
|
if (lua_type(L, 1) == LUA_TSTRING) {
|
|
srv = lua_tostring(L, 1);
|
|
if (srv[0] == '\0')
|
|
srv = NULL;
|
|
}
|
|
else
|
|
srv = NULL;
|
|
|
|
if (lua_type(L, 2) == LUA_TSTRING) {
|
|
chan = lua_tostring(L, 2);
|
|
if (chan[0] == '\0')
|
|
chan = NULL;
|
|
}
|
|
else
|
|
chan = NULL;
|
|
|
|
ptr = xchat_find_context(ph, srv, chan);
|
|
ctx = (long)ptr;
|
|
#ifdef DEBUG
|
|
fprintf(stderr, "find_context(): %#lx\n", (long)ptr);
|
|
#endif
|
|
lua_pushnumber(L, (double)ctx);
|
|
return 1;
|
|
}
|
|
|
|
/*
|
|
* lua: xchat.get_context()
|
|
* desc: Returns the current context for your plugin. You can use this later
|
|
* with xchat_set_context.
|
|
* ret: context number ... DON'T modifiy
|
|
* args: none
|
|
*/
|
|
static int lxc_get_context(lua_State *L)
|
|
{
|
|
long ptr;
|
|
xchat_context *ctx = xchat_get_context(ph);
|
|
ptr = (long)ctx;
|
|
#ifdef DEBUG
|
|
fprintf(stderr, "get_context(): %#lx\n", ptr);
|
|
#endif
|
|
lua_pushnumber(L, (double)ptr);
|
|
return 1;
|
|
}
|
|
|
|
/*
|
|
* lua: xchat.get_info(id)
|
|
* desc: Returns information based on your current context.
|
|
* ret: the requested string or nil on error
|
|
* args:
|
|
* * id (string): the wanted information
|
|
*/
|
|
static int lxc_get_info(lua_State *L)
|
|
{
|
|
const char *id = luaL_checkstring(L, 1);
|
|
const char *value = xchat_get_info(ph, id);
|
|
if (value == NULL)
|
|
lua_pushnil(L);
|
|
else
|
|
lua_pushstring(L, value);
|
|
return 1;
|
|
}
|
|
|
|
/*
|
|
* lua: xchat.get_prefs(name)
|
|
* desc: Provides xchat's setting information (that which is available
|
|
* through the /set command). A few extra bits of information are
|
|
* available that don't appear in the /set list, currently they are:
|
|
* * state_cursor: Current input-box cursor position (characters,
|
|
* not bytes). Since 2.4.2.
|
|
* *id: Unique server id. Since 2.6.1.
|
|
* ret: returns the string/number/boolean for the given config var
|
|
* or nil on error
|
|
* args:
|
|
* * name (string): the wanted setting's name
|
|
*/
|
|
static int lxc_get_prefs(lua_State *L)
|
|
{
|
|
int i;
|
|
const char *str;
|
|
const char *name = luaL_checkstring(L, 1);
|
|
/*
|
|
* luckily we can store anything in a lua var... this makes the
|
|
* xchat lua api more user friendly ;-)
|
|
*/
|
|
switch (xchat_get_prefs(ph, name, &str, &i)) {
|
|
case 0: /* request failed */
|
|
lua_pushnil(L);
|
|
break;
|
|
case 1:
|
|
lua_pushstring(L, str);
|
|
break;
|
|
case 2:
|
|
lua_pushnumber(L, (double)i);
|
|
break;
|
|
case 3:
|
|
lua_pushboolean(L, i);
|
|
break;
|
|
default: /* doesn't happen if xchat's C-API doesn't change ;-) */
|
|
lua_pushnil(L);
|
|
break;
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
/*
|
|
* lua: xchat.set_context(ctx)
|
|
* desc: Changes your current context to the one given.
|
|
* ret: true or false
|
|
* args:
|
|
* * ctx (number): the context (e.g. from xchat.get_context())
|
|
*/
|
|
static int lxc_set_context(lua_State *L)
|
|
{
|
|
double ctx = luaL_checknumber(L, 1);
|
|
#ifdef DEBUG
|
|
fprintf(stderr, "set_context(): %#lx\n", (long)ctx);
|
|
#endif
|
|
xchat_context *xc = (void *)(long)ctx;
|
|
lua_pushboolean(L, xchat_set_context(ph, xc));
|
|
return 1;
|
|
}
|
|
|
|
/*
|
|
* lua: xchat.nickcmp(name1, name2)
|
|
* desc: Performs a nick name comparision, based on the current server
|
|
* connection. This might be a RFC1459 compliant string compare, or
|
|
* plain ascii (in the case of DALNet). Use this to compare channels
|
|
* and nicknames. The function works the same way as strcasecmp.
|
|
* ret: number ess than, equal to, or greater than zero if name1 is found,
|
|
* respectively, to be less than, to match, or be greater than name2.
|
|
* args:
|
|
* * name1 (string): nick or channel name
|
|
* * name2 (string): nick or channel name
|
|
*/
|
|
static int lxc_nickcmp(lua_State *L)
|
|
{
|
|
const char *n1 = luaL_checkstring(L, 1);
|
|
const char *n2 = luaL_checkstring(L, 2);
|
|
lua_pushnumber(L, (double)xchat_nickcmp(ph, n1, n2));
|
|
return 1;
|
|
}
|
|
|
|
/*
|
|
* lua: xchat.list_get(name)
|
|
* desc: http://xchat.org/docs/plugin20.html#lists :)
|
|
* time_t values are stored as number => e.g.
|
|
* os.date("%Y-%m-%d, %H:%M:%S", time_t_value)
|
|
* pointers (channel -> context) as number... untested if this works
|
|
* ret: table with tables with all keys=Name, value=(Type)... or nil on error
|
|
* args:
|
|
* * name (string): the wanted list
|
|
*/
|
|
static int lxc_list_get(lua_State *L)
|
|
{
|
|
const char *name = luaL_checkstring(L, 1);
|
|
int i; /* item index */
|
|
int l; /* list index */
|
|
const char *str;
|
|
double num;
|
|
time_t date;
|
|
long ptr;
|
|
const char *const *fields = xchat_list_fields(ph, name);
|
|
xchat_list *list = xchat_list_get(ph, name);
|
|
|
|
if (!list) {
|
|
lua_pushnil(L);
|
|
return 1;
|
|
}
|
|
lua_newtable(L);
|
|
/* this is like the perl plugin does it ;-) */
|
|
l = 1;
|
|
while (xchat_list_next(ph, list)) {
|
|
i = 0;
|
|
lua_pushnumber(L, l);
|
|
lua_newtable(L);
|
|
while (fields[i] != NULL) {
|
|
switch (fields[i][0]) {
|
|
case 's':
|
|
str = xchat_list_str(ph, list, fields [i] + 1);
|
|
lua_pushstring(L, fields[i]+1);
|
|
if (str != NULL)
|
|
lua_pushstring(L, str);
|
|
else
|
|
lua_pushnil(L);
|
|
lua_settable(L, -3);
|
|
break;
|
|
case 'p':
|
|
ptr = (long)xchat_list_str(ph, list, fields [i] + 1);
|
|
num = (double)ptr;
|
|
lua_pushstring(L, fields[i]+1);
|
|
lua_pushnumber(L, num);
|
|
lua_settable(L, -3);
|
|
break;
|
|
case 'i':
|
|
num = (double)xchat_list_int(ph, list, fields[i] + 1);
|
|
lua_pushstring(L, fields[i]+1);
|
|
lua_pushnumber(L, num);
|
|
lua_settable(L, -3);
|
|
break;
|
|
case 't':
|
|
date = xchat_list_time(ph, list, fields[i] + 1);
|
|
lua_pushstring(L, fields[i]+1);
|
|
lua_pushnumber(L, (double)date);
|
|
lua_settable(L, -3);
|
|
break;
|
|
}
|
|
i++;
|
|
}
|
|
lua_settable(L, -3);
|
|
l++;
|
|
}
|
|
xchat_list_free(ph, list);
|
|
return 1;
|
|
}
|
|
|
|
/*
|
|
* lua: xchat.list_fields(name)
|
|
* desc: returns the possible keys for name as table
|
|
* ret: table ;->
|
|
* args:
|
|
* * name (string): name of the wanted list ("channels", "dcc",
|
|
* "ignore", "notify", "users")
|
|
*/
|
|
static int lxc_list_fields(lua_State *L)
|
|
{
|
|
const char *name = luaL_checkstring(L, 1);
|
|
const char *const *fields = xchat_list_fields(ph, name);
|
|
int i;
|
|
|
|
lua_newtable(L);
|
|
i = 0;
|
|
while (fields[i] != NULL) {
|
|
lua_pushnumber(L, i);
|
|
/* first char is the type ... */
|
|
lua_pushstring(L, fields[i]+1);
|
|
lua_settable(L, -3);
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
/*
|
|
* lua: xchat.gettext(str)
|
|
* desc:
|
|
*/
|
|
static int lxc_gettext(lua_State *L)
|
|
{
|
|
#if defined(_WIN32) || defined(LXC_XCHAT_GETTEXT)
|
|
lua_pushstring(L, xchat_gettext(ph, luaL_checkstring(L, 1)));
|
|
#else
|
|
const char *dom;
|
|
const char *msgid = luaL_checkstring(L, 1);
|
|
if (lua_type(L,2) == LUA_TSTRING)
|
|
dom = lua_tostring(L, 2);
|
|
else
|
|
dom = "xchat";
|
|
lua_pushstring(L, dgettext(dom, msgid));
|
|
#endif
|
|
return 1;
|
|
}
|
|
|
|
/*
|
|
* lua: xchat.bits(flags)
|
|
* desc: returns a table of booleans if the bit at index (err... index-1) is
|
|
* set
|
|
* ret: table of booleans
|
|
* args:
|
|
* * flags (number)
|
|
*/
|
|
static int lxc_bits(lua_State *L)
|
|
{
|
|
int flags = luaL_checknumber(L, 1);
|
|
int i;
|
|
lua_pop(L, 1);
|
|
lua_newtable(L);
|
|
for (i=0; i<16; i++) { /* at time of writing, the highest index was 9 ... */
|
|
lua_pushnumber(L, i+1);
|
|
lua_pushboolean(L, ((1<<i) & flags) == 0 ? 0 : 1);
|
|
lua_settable(L, -3);
|
|
}
|
|
return 1;
|
|
}
|
|
/*
|
|
* vim: ts=3 noexpandtab
|
|
*/
|