1
0
mirror of https://github.com/weechat/weechat.git synced 2026-06-26 04:46:37 +02:00
Files
weechat/src/plugins/scripts/ruby/weechat-ruby.c
T
Emmanuel Bouthenot b8662d79c2 rewrite ruby funcalls
2005-12-10 22:28:18 +00:00

1359 lines
38 KiB
C

/*
* Copyright (c) 2003-2005 by FlashCode <flashcode@flashtux.org>
* See README for License detail, AUTHORS for developers list.
*
* 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; either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU 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
*/
/* wee-ruby.c: Ruby plugin support for WeeChat */
#include <ruby.h>
#include <stdlib.h>
#include <string.h>
#include <stdarg.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#undef _
#include "../../weechat-plugin.h"
#include "../weechat-script.h"
char plugin_name[] = "Ruby";
char plugin_version[] = "0.1";
char plugin_description[] = "Ruby scripts support";
t_weechat_plugin *ruby_plugin;
t_plugin_script *ruby_scripts = NULL;
t_plugin_script *ruby_current_script = NULL;
char *ruby_current_script_filename = NULL;
VALUE mWeechat, mWeechatOutputs;
#define MOD_NAME_PREFIX "WeechatRubyModule"
int modnum = 0;
/*
* protect_funcall0 :
*
*/
static VALUE
rb_funcall0 (VALUE args)
{
VALUE recv = rb_ary_shift (args);
VALUE func = rb_ary_shift (args);
VALUE argc = rb_ary_shift (args);
return rb_funcall2(recv, rb_intern(STR2CSTR(func)), NUM2INT(argc), RARRAY(args)->ptr);
}
/*
* protect_rescue0 :
*
*/
VALUE
rb_rescue0 (VALUE func, VALUE error_info)
{
VALUE str = rb_funcall(error_info, rb_intern("to_s"), 0, NULL);
VALUE errinfo = rb_inspect(ruby_errinfo);
ruby_plugin->printf_server (ruby_plugin,
"Ruby error: unable to run function \"%s\"",
STR2CSTR(func));
ruby_plugin->printf_server (ruby_plugin,
"Ruby error: %s", STR2CSTR(str));
ruby_plugin->printf_server (ruby_plugin,
"Ruby error: %s", STR2CSTR(errinfo));
//rb_backtrace ();
return INT2FIX(PLUGIN_RC_KO);
}
/*
* rb_rescue_funcall :
*
*/
VALUE
rb_rescue_funcall (VALUE recv, VALUE func, int argc, ...)
{
va_list ap;
VALUE argv;
argv = rb_ary_new ();
rb_ary_push (argv, recv);
rb_ary_push (argv, func);
rb_ary_push (argv, INT2FIX(argc));
if (argc > 0)
{
int i;
va_start(ap, argc);
for (i = 0; i < argc; i++)
rb_ary_push (argv, va_arg(ap, VALUE));
va_end(ap);
}
VALUE ret = rb_rescue(rb_funcall0, argv, rb_rescue0, func);
if (NIL_P(ret))
ret = INT2FIX(PLUGIN_RC_KO);
return ret;
}
/*
* weechat_ruby_exec: execute a Ruby script
*/
int
weechat_ruby_exec (t_weechat_plugin *plugin,
t_plugin_script *script,
char *function, char *server, char *arguments)
{
/* make gcc happy */
(void) plugin;
VALUE ruby_retcode;
ruby_current_script = script;
ruby_retcode = rb_rescue_funcall ((VALUE) script->interpreter, rb_str_new2(function), 2,
rb_str_new2((server == NULL) ? "" : server),
rb_str_new2((arguments == NULL) ? "" : arguments));
return NUM2INT(ruby_retcode);
}
/*
* weechat_ruby_handler: general message and command handler for Ruby
*/
int
weechat_ruby_handler (t_weechat_plugin *plugin,
char *server, char *command, char *arguments,
char *handler_args, void *handler_pointer)
{
/* make gcc happy */
(void) command;
return weechat_ruby_exec (plugin, (t_plugin_script *)handler_pointer,
handler_args, server, arguments);
}
/*
* weechat_ruby_register: startup function for all WeeChat Ruby scripts
*/
static VALUE
weechat_ruby_register (VALUE class, VALUE name, VALUE version,
VALUE shutdown_func, VALUE description)
{
char *c_name, *c_version, *c_shutdown_func, *c_description;
/* make gcc happy */
(void) class;
if (NIL_P (name) || NIL_P (version) || NIL_P (shutdown_func) || NIL_P (description))
{
ruby_plugin->printf_server (ruby_plugin,
"Ruby error: wrong parameters for "
"\"register\" function");
return INT2FIX (0);
}
Check_Type (name, T_STRING);
Check_Type (version, T_STRING);
Check_Type (shutdown_func, T_STRING);
Check_Type (description, T_STRING);
c_name = STR2CSTR (name);
c_version = STR2CSTR (version);
c_shutdown_func = STR2CSTR (shutdown_func);
c_description = STR2CSTR (description);
if (weechat_script_search (ruby_plugin, &ruby_scripts, c_name))
{
/* error: another scripts already exists with this name! */
ruby_plugin->printf_server (ruby_plugin,
"Ruby error: unable to register "
"\"%s\" script (another script "
"already exists with this name)",
c_name);
return INT2FIX (0);
}
/* register script */
ruby_current_script = weechat_script_add (ruby_plugin,
&ruby_scripts,
(ruby_current_script_filename) ?
ruby_current_script_filename : "",
c_name, c_version, c_shutdown_func,
c_description);
if (ruby_current_script)
{
ruby_plugin->printf_server (ruby_plugin,
"Ruby: registered script \"%s\", "
"version %s (%s)",
c_name, c_version, c_description);
}
else
{
ruby_plugin->printf_server (ruby_plugin,
"Ruby error: unable to load script "
"\"%s\" (not enough memory)",
c_name);
return INT2FIX (0);
}
return INT2FIX (1);
}
/*
* weechat_ruby_print: print message into a buffer (current or specified one)
*/
static VALUE
weechat_ruby_print (int argc, VALUE *argv, VALUE class)
{
VALUE message, channel_name, server_name;
char *c_message, *c_channel_name, *c_server_name;
/* make gcc happy */
(void) class;
if (!ruby_current_script)
{
ruby_plugin->printf_server (ruby_plugin,
"Ruby error: unable to print message, "
"script not initialized");
return INT2FIX (0);
}
message = Qnil;
channel_name = Qnil;
server_name = Qnil;
c_message = NULL;
c_channel_name = NULL;
c_server_name = NULL;
rb_scan_args (argc, argv, "12", &message, &channel_name, &server_name);
if (NIL_P (message))
{
ruby_plugin->printf_server (ruby_plugin,
"Ruby error: wrong parameters for "
"\"print\" function");
return INT2FIX (0);
}
Check_Type (message, T_STRING);
c_message = STR2CSTR (message);
if (!NIL_P (channel_name))
{
Check_Type (channel_name, T_STRING);
c_channel_name = STR2CSTR (channel_name);
}
if (!NIL_P (server_name))
{
Check_Type (server_name, T_STRING);
c_server_name = STR2CSTR (server_name);
}
ruby_plugin->printf (ruby_plugin,
c_server_name, c_channel_name,
"%s", c_message);
return INT2FIX (1);
}
/*
* weechat_ruby_print_infobar: print message to infobar
*/
static VALUE
weechat_ruby_print_infobar (VALUE class, VALUE delay, VALUE message)
{
int c_delay;
char *c_message;
/* make gcc happy */
(void) class;
if (!ruby_current_script)
{
ruby_plugin->printf_server (ruby_plugin,
"Ruby error: unable to print infobar message, "
"script not initialized");
return INT2FIX (0);
}
c_delay = 1;
c_message = NULL;
if (NIL_P (delay) || NIL_P (message))
{
ruby_plugin->printf_server (ruby_plugin,
"Ruby error: wrong parameters for "
"\"print_infobar\" function");
return INT2FIX (0);
}
Check_Type (delay, T_FIXNUM);
Check_Type (message, T_STRING);
c_delay = FIX2INT (delay);
c_message = STR2CSTR (message);
ruby_plugin->infobar_printf (ruby_plugin, c_delay, c_message);
return INT2FIX (1);
}
/*
* weechat_ruby_command: send command to server
*/
static VALUE
weechat_ruby_command (int argc, VALUE *argv, VALUE class)
{
VALUE command, channel_name, server_name;
char *c_command, *c_channel_name, *c_server_name;
/* make gcc happy */
(void) class;
if (!ruby_current_script)
{
ruby_plugin->printf_server (ruby_plugin,
"Ruby error: unable to run command, "
"script not initialized");
return INT2FIX (0);
}
command = Qnil;
channel_name = Qnil;
server_name = Qnil;
c_command = NULL;
c_channel_name = NULL;
c_server_name = NULL;
rb_scan_args(argc, argv, "12", &command, &channel_name, &server_name);
if (NIL_P (command))
{
ruby_plugin->printf_server (ruby_plugin,
"Ruby error: wrong parameters for "
"\"command\" function");
return INT2FIX (0);
}
Check_Type (command, T_STRING);
c_command = STR2CSTR (command);
if (!NIL_P (channel_name))
{
Check_Type (channel_name, T_STRING);
c_channel_name = STR2CSTR (channel_name);
}
if (!NIL_P (server_name))
{
Check_Type (server_name, T_STRING);
c_server_name = STR2CSTR (server_name);
}
ruby_plugin->exec_command (ruby_plugin,
c_server_name, c_channel_name,
c_command);
return INT2FIX (1);
}
/*
* weechat_ruby_add_message_handler: add handler for messages
*/
static VALUE
weechat_ruby_add_message_handler (VALUE class, VALUE message, VALUE function)
{
char *c_message, *c_function;
/* make gcc happy */
(void) class;
if (!ruby_current_script)
{
ruby_plugin->printf_server (ruby_plugin,
"Ruby error: unable to add message handler, "
"script not initialized");
return INT2FIX (0);
}
c_message = NULL;
c_function = NULL;
if (NIL_P (message) || NIL_P (function))
{
ruby_plugin->printf_server (ruby_plugin,
"Ruby error: wrong parameters for "
"\"add_message_handler\" function");
return INT2FIX (0);
}
Check_Type (message, T_STRING);
Check_Type (function, T_STRING);
c_message = STR2CSTR (message);
c_function = STR2CSTR (function);
if (ruby_plugin->msg_handler_add (ruby_plugin, c_message,
weechat_ruby_handler, c_function,
(void *)ruby_current_script))
return INT2FIX (1);
return INT2FIX (0);
}
/*
* weechat_add_command_handler: define/redefines commands
*/
static VALUE
weechat_ruby_add_command_handler (int argc, VALUE *argv, VALUE class)
{
VALUE command, function, description, arguments, arguments_description;
char *c_command, *c_function, *c_description, *c_arguments, *c_arguments_description;
/* make gcc happy */
(void) class;
if (!ruby_current_script)
{
ruby_plugin->printf_server (ruby_plugin,
"Ruby error: unable to add command handler, "
"script not initialized");
return INT2FIX (0);
}
command = Qnil;
function = Qnil;
description = Qnil;
arguments = Qnil;
arguments_description = Qnil;
c_command = NULL;
c_function = NULL;
c_description = NULL;
c_arguments = NULL;
c_arguments_description = NULL;
rb_scan_args (argc, argv, "23", &command, &function, &description,
&arguments, &arguments_description);
if (NIL_P (command) || NIL_P (function))
{
ruby_plugin->printf_server (ruby_plugin,
"Ruby error: wrong parameters for "
"\"add_command_handler\" function");
return INT2FIX (0);
}
Check_Type (command, T_STRING);
c_command = STR2CSTR (command);
Check_Type (function, T_STRING);
c_function = STR2CSTR (function);
if (!NIL_P (description))
{
Check_Type (description, T_STRING);
c_description = STR2CSTR (description);
}
if (!NIL_P (arguments))
{
Check_Type (arguments, T_STRING);
c_arguments = STR2CSTR (arguments);
}
if (!NIL_P (arguments_description)) {
Check_Type (arguments_description, T_STRING);
c_arguments_description = STR2CSTR (arguments_description);
}
if (ruby_plugin->cmd_handler_add (ruby_plugin,
c_command,
c_description,
c_arguments,
c_arguments_description,
weechat_ruby_handler,
c_function,
(void *)ruby_current_script))
return INT2FIX (1);
return INT2FIX (0);
}
/*
* weechat_remove_handler: remove a handler
*/
static VALUE
weechat_ruby_remove_handler (VALUE class, VALUE command, VALUE function)
{
char *c_command, *c_function;
/* make gcc happy */
(void) class;
if (!ruby_current_script)
{
ruby_plugin->printf_server (ruby_plugin,
"Ruby error: unable to remove handler, "
"script not initialized");
return INT2FIX (0);
}
c_command = NULL;
c_function = NULL;
if (NIL_P (command) || NIL_P (function))
{
ruby_plugin->printf_server (ruby_plugin,
"Ruby error: wrong parameters for "
"\"remove_handler\" function");
return INT2FIX (0);
}
Check_Type (command, T_STRING);
Check_Type (function, T_STRING);
c_command = STR2CSTR (command);
c_function = STR2CSTR (function);
weechat_script_remove_handler (ruby_plugin, ruby_current_script,
c_command, c_function);
return INT2FIX (1);
}
/*
* weechat_ruby_get_info: get various infos
*/
static VALUE
weechat_ruby_get_info (int argc, VALUE *argv, VALUE class)
{
char *c_arg, *c_server_name, *info;
VALUE arg, server_name, return_value;
/* make gcc happy */
(void) class;
if (!ruby_current_script)
{
ruby_plugin->printf_server (ruby_plugin,
"Ruby error: unable to get info, "
"script not initialized");
return INT2FIX (0);
}
arg = Qnil;
server_name = Qnil;
c_arg = NULL;
c_server_name = NULL;
rb_scan_args (argc, argv, "11", &arg, &server_name);
if (NIL_P (arg))
{
ruby_plugin->printf_server (ruby_plugin,
"Ruby error: wrong parameters for "
"\"get_info\" function");
return INT2FIX (0);
}
Check_Type (arg, T_STRING);
c_arg = STR2CSTR (arg);
if (!NIL_P (server_name))
{
Check_Type (server_name, T_STRING);
c_server_name = STR2CSTR (server_name);
}
if (c_arg)
{
info = ruby_plugin->get_info (ruby_plugin, c_arg, c_server_name);
if (info)
{
return_value = rb_str_new2 (info);
free (info);
return return_value;
}
}
return rb_str_new2 ("");
}
/*
* weechat_ruby_get_dcc_info: get infos about DCC
*/
static VALUE
weechat_ruby_get_dcc_info (VALUE class)
{
t_plugin_dcc_info *dcc_info, *ptr_dcc;
VALUE dcc_list, dcc_list_member;
char timebuffer1[64];
char timebuffer2[64];
struct in_addr in;
/* make gcc happy */
(void) class;
if (!ruby_current_script)
{
ruby_plugin->printf_server (ruby_plugin,
"Ruby error: unable to get DCC info, "
"script not initialized");
return INT2FIX (0);
}
dcc_list = rb_ary_new();
if (NIL_P (dcc_list))
return Qnil;
dcc_info = ruby_plugin->get_dcc_info (ruby_plugin);
if (!dcc_info)
return dcc_list;
for(ptr_dcc = dcc_info; ptr_dcc; ptr_dcc = ptr_dcc->next_dcc)
{
strftime(timebuffer1, sizeof(timebuffer1), "%F %T",
localtime(&ptr_dcc->start_time));
strftime(timebuffer2, sizeof(timebuffer2), "%F %T",
localtime(&ptr_dcc->start_transfer));
in.s_addr = htonl(ptr_dcc->addr);
dcc_list_member = rb_hash_new ();
if (!NIL_P (dcc_list_member))
{
rb_hash_aset (dcc_list_member, rb_str_new2("server"),
rb_str_new2(ptr_dcc->server));
rb_hash_aset (dcc_list_member, rb_str_new2("channel"),
rb_str_new2(ptr_dcc->channel));
rb_hash_aset (dcc_list_member, rb_str_new2("type"),
INT2FIX(ptr_dcc->type));
rb_hash_aset (dcc_list_member, rb_str_new2("status"),
INT2FIX(ptr_dcc->status));
rb_hash_aset (dcc_list_member, rb_str_new2("start_time"),
rb_str_new2(timebuffer1));
rb_hash_aset (dcc_list_member, rb_str_new2("start_transfer"),
rb_str_new2(timebuffer2));
rb_hash_aset (dcc_list_member, rb_str_new2("address"),
rb_str_new2(inet_ntoa(in)));
rb_hash_aset (dcc_list_member, rb_str_new2("port"),
INT2FIX(ptr_dcc->port));
rb_hash_aset (dcc_list_member, rb_str_new2("nick"),
rb_str_new2(ptr_dcc->nick));
rb_hash_aset (dcc_list_member, rb_str_new2("remote_file"),
rb_str_new2(ptr_dcc->filename));
rb_hash_aset (dcc_list_member, rb_str_new2("local_file"),
rb_str_new2(ptr_dcc->local_filename));
rb_hash_aset (dcc_list_member, rb_str_new2("filename_suffix"),
INT2FIX(ptr_dcc->filename_suffix));
rb_hash_aset (dcc_list_member, rb_str_new2("size"),
INT2FIX(ptr_dcc->size));
rb_hash_aset (dcc_list_member, rb_str_new2("pos"),
INT2FIX(ptr_dcc->pos));
rb_hash_aset (dcc_list_member, rb_str_new2("start_resume"),
INT2FIX(ptr_dcc->start_resume));
rb_hash_aset (dcc_list_member, rb_str_new2("cps"),
INT2FIX(ptr_dcc->bytes_per_sec));
if (NIL_P (rb_ary_push (dcc_list, dcc_list_member)))
{
rb_gc_unregister_address (&dcc_list_member);
rb_gc_unregister_address (&dcc_list);
ruby_plugin->free_dcc_info (ruby_plugin, dcc_info);
return Qnil;
}
}
else
{
rb_gc_unregister_address (&dcc_list);
ruby_plugin->free_dcc_info (ruby_plugin, dcc_info);
return Qnil;
}
}
ruby_plugin->free_dcc_info (ruby_plugin, dcc_info);
return dcc_list;
}
/*
* weechat_ruby_get_config: get value of a WeeChat config option
*/
static VALUE
weechat_ruby_get_config (VALUE class, VALUE option)
{
char *c_option, *return_value;
VALUE ruby_return_value;
/* make gcc happy */
(void) class;
if (!ruby_current_script)
{
ruby_plugin->printf_server (ruby_plugin,
"Ruby error: unable to get config option, "
"script not initialized");
return INT2FIX (0);
}
c_option = NULL;
if (NIL_P (option))
{
ruby_plugin->printf_server (ruby_plugin,
"Ruby error: wrong parameters for "
"\"get_config\" function");
return INT2FIX (0);
}
Check_Type (option, T_STRING);
c_option = STR2CSTR (option);
if (c_option)
{
return_value = ruby_plugin->get_config (ruby_plugin, c_option);
if (return_value)
{
ruby_return_value = rb_str_new2 (return_value);
free (return_value);
return ruby_return_value;
}
}
return rb_str_new2 ("");
}
/*
* weechat_ruby_set_config: set value of a WeeChat config option
*/
static VALUE
weechat_ruby_set_config (VALUE class, VALUE option, VALUE value)
{
char *c_option, *c_value;
/* make gcc happy */
(void) class;
if (!ruby_current_script)
{
ruby_plugin->printf_server (ruby_plugin,
"Ruby error: unable to set config option, "
"script not initialized");
return INT2FIX (0);
}
c_option = NULL;
c_value = NULL;
if (NIL_P (option))
{
ruby_plugin->printf_server (ruby_plugin,
"Ruby error: wrong parameters for "
"\"set_config\" function");
return INT2FIX (0);
}
Check_Type (option, T_STRING);
Check_Type (value, T_STRING);
c_option = STR2CSTR (option);
c_value = STR2CSTR (value);
if (c_option && c_value)
{
if (ruby_plugin->set_config (ruby_plugin, c_option, c_value))
return INT2FIX (1);
}
return INT2FIX (0);
}
/*
* weechat_ruby_get_plugin_config: get value of a plugin config option
*/
static VALUE
weechat_ruby_get_plugin_config (VALUE class, VALUE option)
{
char *c_option, *return_value;
VALUE ruby_return_value;
/* make gcc happy */
(void) class;
if (!ruby_current_script)
{
ruby_plugin->printf_server (ruby_plugin,
"Ruby error: unable to get plugin config option, "
"script not initialized");
return INT2FIX (0);
}
c_option = NULL;
if (NIL_P (option))
{
ruby_plugin->printf_server (ruby_plugin,
"Ruby error: wrong parameters for "
"\"get_plugin_config\" function");
return INT2FIX (0);
}
Check_Type (option, T_STRING);
c_option = STR2CSTR (option);
if (c_option)
{
return_value = weechat_script_get_plugin_config (ruby_plugin,
ruby_current_script,
c_option);
if (return_value)
{
ruby_return_value = rb_str_new2 (return_value);
free (return_value);
return ruby_return_value;
}
}
return rb_str_new2 ("");
}
/*
* weechat_ruby_set_plugin_config: set value of a plugin config option
*/
static VALUE
weechat_ruby_set_plugin_config (VALUE class, VALUE option, VALUE value)
{
char *c_option, *c_value;
/* make gcc happy */
(void) class;
if (!ruby_current_script)
{
ruby_plugin->printf_server (ruby_plugin,
"Ruby error: unable to set plugin config option, "
"script not initialized");
return INT2FIX (0);
}
c_option = NULL;
c_value = NULL;
if (NIL_P (option))
{
ruby_plugin->printf_server (ruby_plugin,
"Ruby error: wrong parameters for "
"\"set_plugin_config\" function");
return INT2FIX (0);
}
Check_Type (option, T_STRING);
Check_Type (value, T_STRING);
c_option = STR2CSTR (option);
c_value = STR2CSTR (value);
if (c_option && c_value)
{
if (weechat_script_set_plugin_config (ruby_plugin,
ruby_current_script,
c_option, c_value))
return INT2FIX (1);
}
return INT2FIX (0);
}
/*
* weechat_ruby_output : redirection for stdout and stderr
*/
static VALUE
weechat_ruby_output(VALUE self, VALUE str)
{
/* make gcc happy */
(void) self;
ruby_plugin->printf_server (ruby_plugin,
"Ruby stdout/stderr: %s",
STR2CSTR(str));
return Qnil;
}
/*
* weechat_ruby_load: load a Ruby script
*/
int
weechat_ruby_load (t_weechat_plugin *plugin, char *filename)
{
char modname[64];
VALUE curModule;
VALUE ruby_retcode;
plugin->printf_server (plugin, "Loading Ruby script \"%s\"", filename);
ruby_current_script = NULL;
snprintf(modname, sizeof(modname), "%s%d", MOD_NAME_PREFIX, modnum);
modnum++;
curModule = rb_define_module(modname);
ruby_current_script_filename = strdup (filename);
ruby_retcode = rb_rescue_funcall (curModule, rb_str_new2("load_eval_file"),
1, rb_str_new2(filename));
free (ruby_current_script_filename);
if (NUM2INT(ruby_retcode) != 0)
{
VALUE ruby_eval_error;
switch (NUM2INT(ruby_retcode))
{
case 1:
ruby_plugin->printf_server (ruby_plugin,
"Ruby error: unable to read file \"%s\"",
filename);
break;
case 2:
ruby_plugin->printf_server (ruby_plugin,
"Ruby error: error while loading file \"%s\"",
filename);
break;
case 3:
ruby_plugin->printf_server (ruby_plugin,
"Ruby error: unable to find \"weechat_init\" function in file \"%s\"",
filename);
break;
}
if (NUM2INT(ruby_retcode) == 1 || NUM2INT(ruby_retcode) == 2)
{
ruby_eval_error = rb_iv_get(curModule, "@load_eval_file_error");
if (ruby_eval_error)
ruby_plugin->printf_server (ruby_plugin,
"Ruby error: %s", STR2CSTR(ruby_eval_error));
}
return 0;
}
ruby_retcode = rb_rescue_funcall (curModule, rb_str_new2("weechat_init"), 0);
if (NUM2INT(ruby_retcode) != PLUGIN_RC_OK)
{
ruby_plugin->printf_server (ruby_plugin,
"Ruby error: unable to eval weechat_init in file \"%s\"",
filename);
if (ruby_current_script != NULL)
weechat_script_remove (plugin, &ruby_scripts, ruby_current_script);
return 0;
}
if (ruby_current_script == NULL) {
plugin->printf_server (plugin,
"Ruby error: function \"register\" not found "
"in file \"%s\"",
filename);
return 0;
}
ruby_current_script->interpreter = (VALUE *) curModule;
rb_gc_register_address (ruby_current_script->interpreter);
return 1;
}
/*
* weechat_ruby_unload: unload a Ruby script
*/
void
weechat_ruby_unload (t_weechat_plugin *plugin, t_plugin_script *script)
{
plugin->printf_server (plugin,
"Unloading Ruby script \"%s\"",
script->name);
if (script->interpreter)
rb_gc_unregister_address (script->interpreter);
weechat_script_remove (plugin, &ruby_scripts, script);
}
/*
* weechat_ruby_unload_name: unload a Ruby script by name
*/
void
weechat_ruby_unload_name (t_weechat_plugin *plugin, char *name)
{
t_plugin_script *ptr_script;
ptr_script = weechat_script_search (plugin, &ruby_scripts, name);
if (ptr_script)
{
weechat_ruby_unload (plugin, ptr_script);
plugin->printf_server (plugin,
"Ruby script \"%s\" unloaded",
name);
}
else
{
plugin->printf_server (plugin,
"Ruby error: script \"%s\" not loaded",
name);
}
}
/*
* weechat_ruby_unload_all: unload all Ruby scripts
*/
void
weechat_ruby_unload_all (t_weechat_plugin *plugin)
{
plugin->printf_server (plugin,
"Unloading all Ruby scripts");
while (ruby_scripts)
weechat_ruby_unload (plugin, ruby_scripts);
plugin->printf_server (plugin,
"Ruby scripts unloaded");
}
/*
* weechat_ruby_cmd: /ruby command handler
*/
int
weechat_ruby_cmd (t_weechat_plugin *plugin,
char *server, char *command, char *arguments,
char *handler_args, void *handler_pointer)
{
int argc, path_length, handler_found;
char **argv, *path_script, *dir_home;
t_plugin_script *ptr_script;
t_plugin_handler *ptr_handler;
/* make gcc happy */
(void) server;
(void) command;
(void) handler_args;
(void) handler_pointer;
if (arguments)
argv = plugin->explode_string (plugin, arguments, " ", 0, &argc);
else
{
argv = NULL;
argc = 0;
}
switch (argc)
{
case 0:
/* list registered Ruby scripts */
plugin->printf_server (plugin, "");
plugin->printf_server (plugin, "Registered Ruby scripts:");
if (ruby_scripts)
{
for (ptr_script = ruby_scripts;
ptr_script; ptr_script = ptr_script->next_script)
{
plugin->printf_server (plugin, " %s v%s%s%s",
ptr_script->name,
ptr_script->version,
(ptr_script->description[0]) ? " - " : "",
ptr_script->description);
}
}
else
plugin->printf_server (plugin, " (none)");
/* list Ruby message handlers */
plugin->printf_server (plugin, "");
plugin->printf_server (plugin, "Ruby message handlers:");
handler_found = 0;
for (ptr_handler = plugin->handlers;
ptr_handler; ptr_handler = ptr_handler->next_handler)
{
if ((ptr_handler->type == HANDLER_MESSAGE)
&& (ptr_handler->handler_args))
{
handler_found = 1;
plugin->printf_server (plugin, " IRC(%s) => Ruby(%s)",
ptr_handler->irc_command,
ptr_handler->handler_args);
}
}
if (!handler_found)
plugin->printf_server (plugin, " (none)");
/* list Ruby command handlers */
plugin->printf_server (plugin, "");
plugin->printf_server (plugin, "Ruby command handlers:");
handler_found = 0;
for (ptr_handler = plugin->handlers;
ptr_handler; ptr_handler = ptr_handler->next_handler)
{
if ((ptr_handler->type == HANDLER_COMMAND)
&& (ptr_handler->handler_args))
{
handler_found = 1;
plugin->printf_server (plugin, " /%s => Ruby(%s)",
ptr_handler->command,
ptr_handler->handler_args);
}
}
if (!handler_found)
plugin->printf_server (plugin, " (none)");
break;
case 1:
if (plugin->ascii_strcasecmp (plugin, argv[0], "autoload") == 0)
weechat_script_auto_load (plugin, "ruby", weechat_ruby_load);
else if (plugin->ascii_strcasecmp (plugin, argv[0], "reload") == 0)
{
weechat_ruby_unload_all (plugin);
weechat_script_auto_load (plugin, "ruby", weechat_ruby_load);
}
else if (plugin->ascii_strcasecmp (plugin, argv[0], "unload") == 0)
weechat_ruby_unload_all (plugin);
break;
case 2:
if (plugin->ascii_strcasecmp (plugin, argv[0], "load") == 0)
{
/* load Ruby script */
if ((strstr (argv[1], "/")) || (strstr (argv[1], "\\")))
path_script = NULL;
else
{
dir_home = plugin->get_info (plugin, "weechat_dir", NULL);
if (dir_home)
{
path_length = strlen (dir_home) + strlen (argv[1]) + 16;
path_script = (char *) malloc (path_length * sizeof (char));
if (path_script)
snprintf (path_script, path_length, "%s/ruby/%s",
dir_home, argv[1]);
else
path_script = NULL;
free (dir_home);
}
else
path_script = NULL;
}
weechat_ruby_load (plugin, (path_script) ? path_script : argv[1]);
if (path_script)
free (path_script);
}
else if (plugin->ascii_strcasecmp (plugin, argv[0], "unload") == 0)
{
/* unload Ruby script */
weechat_ruby_unload_name (plugin, argv[1]);
}
else
{
plugin->printf_server (plugin,
"Ruby error: unknown option for "
"\"ruby\" command");
}
break;
default:
plugin->printf_server (plugin,
"Ruby error: wrong argument count for \"ruby\" command");
}
if (argv)
plugin->free_exploded_string (plugin, argv);
return 1;
}
/*
* weechat_plugin_init: initialize Ruby plugin
*/
int
weechat_plugin_init (t_weechat_plugin *plugin)
{
char *weechat_ruby_code =
{
"class IO\n"
" def write(msg)\n"
" msg.each {|s|\n"
" if (s.chomp != \"\")\n"
" WeechatOutputs.write(msg.chomp)\n"
" end\n"
" }\n"
" end\n"
"end\n"
"\n"
"class Module\n"
" @load_eval_file_error = ''\n"
"\n"
" def load_eval_file (file)\n"
" lines = ''\n"
" begin\n"
" f = File.open(file, 'r')\n"
" lines = f.readlines.join\n"
" rescue => e\n"
" @load_eval_file_error = e\n"
" return 1\n"
" end\n"
"\n"
" begin\n"
" module_eval(lines)\n"
" rescue => e\n"
" @load_eval_file_error = e\n"
" return 2\n"
" end\n"
"\n"
" has_init = false\n"
"\n"
" instance_methods.each do |meth|\n"
" if meth == 'weechat_init'\n"
" has_init = true\n"
" end\n"
" module_eval('module_function :' + meth)\n"
" end\n"
"\n"
" unless has_init\n"
" return 3\n"
" end\n"
"\n"
" return 0\n"
" end\n"
"end\n"
};
ruby_plugin = plugin;
int ruby_error = 0;
plugin->printf_server (plugin, "Loading Ruby module \"weechat\"");
ruby_init ();
ruby_init_loadpath ();
ruby_script ("__weechat_plugin__");
mWeechat = rb_define_module("Weechat");
rb_define_const(mWeechat, "PLUGIN_RC_OK", INT2NUM(PLUGIN_RC_OK));
rb_define_const(mWeechat, "PLUGIN_RC_KO", INT2NUM(PLUGIN_RC_KO));
rb_define_const(mWeechat, "PLUGIN_RC_OK_IGNORE_WEECHAT", INT2NUM(PLUGIN_RC_OK_IGNORE_WEECHAT));
rb_define_const(mWeechat, "PLUGIN_RC_OK_IGNORE_PLUGINS", INT2NUM(PLUGIN_RC_OK_IGNORE_PLUGINS));
rb_define_const(mWeechat, "PLUGIN_RC_OK_IGNORE_ALL", INT2NUM(PLUGIN_RC_OK_IGNORE_ALL));
rb_define_module_function (mWeechat, "register", weechat_ruby_register, 4);
rb_define_module_function (mWeechat, "print", weechat_ruby_print, -1);
rb_define_module_function (mWeechat, "print_infobar", weechat_ruby_print_infobar, 2);
rb_define_module_function (mWeechat, "command", weechat_ruby_command, -1);
rb_define_module_function (mWeechat, "add_message_handler", weechat_ruby_add_message_handler, 2);
rb_define_module_function (mWeechat, "add_command_handler", weechat_ruby_add_command_handler, -1);
rb_define_module_function (mWeechat, "remove_handler", weechat_ruby_remove_handler, 2);
rb_define_module_function (mWeechat, "get_info", weechat_ruby_get_info, -1);
rb_define_module_function (mWeechat, "get_dcc_info", weechat_ruby_get_dcc_info, 0);
rb_define_module_function (mWeechat, "get_config", weechat_ruby_get_config, 1);
rb_define_module_function (mWeechat, "set_config", weechat_ruby_set_config, 2);
rb_define_module_function (mWeechat, "get_plugin_config", weechat_ruby_get_plugin_config, 1);
rb_define_module_function (mWeechat, "set_plugin_config", weechat_ruby_set_plugin_config, 2);
/* redirect stdin and stdout */
mWeechatOutputs = rb_define_module("WeechatOutputs");
rb_define_singleton_method(mWeechatOutputs, "write", weechat_ruby_output, 1);
plugin->cmd_handler_add (plugin, "ruby",
"list/load/unload Ruby scripts",
"[load filename] | [autoload] | [reload] | [unload]",
"filename: Ruby script (file) to load\n\n"
"Without argument, /ruby command lists all loaded Ruby scripts.",
weechat_ruby_cmd, NULL, NULL);
plugin->mkdir_home (plugin, "ruby");
plugin->mkdir_home (plugin, "ruby/autoload");
rb_eval_string_protect(weechat_ruby_code, &ruby_error);
if (ruby_error) {
VALUE ruby_error_info = rb_inspect(ruby_errinfo);
rb_backtrace();
ruby_plugin->printf_server (ruby_plugin,
"Ruby error: unable to eval weechat ruby internal code");
ruby_plugin->printf_server (ruby_plugin,
"Ruby error: %s", STR2CSTR(ruby_error_info));
return PLUGIN_RC_KO;
}
weechat_script_auto_load (plugin, "ruby", weechat_ruby_load);
/* init ok */
return PLUGIN_RC_OK;
}
/*
* weechat_plugin_end: shutdown Ruby interface
*/
void
weechat_plugin_end (t_weechat_plugin *plugin)
{
/* unload all scripts */
weechat_ruby_unload_all (plugin);
ruby_finalize();
ruby_plugin->printf_server (ruby_plugin,
"Ruby plugin ended");
}