1
0
mirror of https://github.com/rkd77/elinks.git synced 2025-01-03 14:57:44 -05:00
elinks/src/scripting/ruby/hooks.c

258 lines
5.4 KiB
C
Raw Normal View History

/* Ruby scripting hooks */
#ifdef HAVE_CONFIG_H
#include "config.h"
#endif
2022-01-15 11:56:00 -05:00
#undef _GNU_SOURCE
#include <ruby.h>
#include "elinks.h"
#include "cache/cache.h"
#include "main/event.h"
#include "protocol/uri.h"
#include "scripting/ruby/core.h"
#include "scripting/ruby/hooks.h"
#include "session/location.h"
#include "session/session.h"
#include "util/string.h"
/* The events that will trigger the functions below and what they are expected
* to do is explained in doc/events.txt */
/* We need to catch and handle errors because, otherwise, Ruby will kill us. */
struct erb_protect_info {
const char *name;
int argc;
VALUE *args;
};
static VALUE
do_erb_protected_method_call(VALUE data)
{
struct erb_protect_info *info = (struct erb_protect_info *) data;
ID method_id;
assert(info);
method_id = rb_intern(info->name);
return rb_funcall3(erb_module, method_id, info->argc, info->args);
}
static VALUE
erb_protected_method_call(const char *name, int argc, VALUE *args, int *error)
{
struct erb_protect_info info = { name, argc, args };
return rb_protect(do_erb_protected_method_call, (VALUE) &info, error);
}
static enum evhook_status
script_hook_goto_url(va_list ap, void *data)
{
char **url = va_arg(ap, char **);
struct session *ses = va_arg(ap, struct session *);
int error;
VALUE args[2];
VALUE result;
if (*url == NULL)
return EVENT_HOOK_STATUS_NEXT;
args[0] = rb_str_new(*url, strlen(*url));
if (!ses || !have_location(ses)) {
args[1] = Qnil;
} else {
args[1] = rb_str_new(struri(cur_loc(ses)->vs.uri), strlen(struri(cur_loc(ses)->vs.uri)));
}
result = erb_protected_method_call("goto_url_hook", 2, args, &error);
if (error) {
erb_report_error(ses, error);
return EVENT_HOOK_STATUS_NEXT;
}
switch (rb_type(result)) {
case T_STRING:
{
char *new_url;
new_url = memacpy(RSTRING_PTR(result), RSTRING_LEN(result));
if (new_url) {
mem_free_set(url, new_url);
}
break;
}
case T_NIL:
break;
default:
alert_ruby_error(ses, "goto_url_hook must return a string or nil");
}
return EVENT_HOOK_STATUS_NEXT;
}
static enum evhook_status
script_hook_follow_url(va_list ap, void *data)
{
char **url = va_arg(ap, char **);
struct session *ses = va_arg(ap, struct session *);
int error;
VALUE args[1];
VALUE result;
evhook_use_params(url && ses);
if (*url == NULL)
return EVENT_HOOK_STATUS_NEXT;
args[0] = rb_str_new(*url, strlen(*url));
result = erb_protected_method_call("follow_url_hook", 1, args, &error);
if (error) {
erb_report_error(ses, error);
return EVENT_HOOK_STATUS_NEXT;
}
switch (rb_type(result)) {
case T_STRING:
{
char *new_url;
new_url = memacpy(RSTRING_PTR(result), RSTRING_LEN(result));
if (new_url) {
mem_free_set(url, new_url);
}
break;
}
case T_NIL:
break;
default:
alert_ruby_error(ses, "follow_url_hook must return a string or nil");
}
return EVENT_HOOK_STATUS_NEXT;
}
static enum evhook_status
script_hook_pre_format_html(va_list ap, void *data)
{
struct session *ses = va_arg(ap, struct session *);
struct cache_entry *cached = va_arg(ap, struct cache_entry *);
struct fragment *fragment = get_cache_fragment(cached);
char *url = struri(cached->uri);
int error;
VALUE args[2];
VALUE result;
evhook_use_params(ses && cached);
if (!cached->length || !*fragment->data)
return EVENT_HOOK_STATUS_NEXT;
args[0] = rb_str_new2(url);
args[1] = rb_str_new(fragment->data, fragment->length);
result = erb_protected_method_call("pre_format_html_hook", 2, args, &error);
if (error) {
erb_report_error(ses, error);
return EVENT_HOOK_STATUS_NEXT;
}
switch (rb_type(result)) {
case T_STRING:
{
int len = RSTRING_LEN(result);
add_fragment(cached, 0, RSTRING_PTR(result), len);
normalize_cache_entry(cached, len);
break;
}
case T_NIL:
break;
default:
alert_ruby_error(ses, "pre_format_html_hook must return a string or nil");
}
return EVENT_HOOK_STATUS_NEXT;
}
/* The Ruby function can return:
* - "PROXY:PORT" to use the specified proxy
* - "" to not use any proxy
* - nil to use the default proxies */
static enum evhook_status
script_hook_get_proxy(va_list ap, void *data)
{
char **new_proxy_url = va_arg(ap, char **);
char *url = va_arg(ap, char *);
int error;
VALUE args[1];
VALUE result;
if (!new_proxy_url || !url)
return EVENT_HOOK_STATUS_NEXT;
args[0] = rb_str_new(url, strlen(url));
result = erb_protected_method_call("proxy_hook", 1, args, &error);
if (error) {
erb_report_error(NULL, error);
return EVENT_HOOK_STATUS_NEXT;
}
switch (rb_type(result)) {
case T_STRING:
{
char *proxy;
proxy = memacpy(RSTRING_PTR(result), RSTRING_LEN(result));
if (proxy) {
mem_free_set(new_proxy_url, proxy);
}
break;
}
case T_NIL:
break;
default:
alert_ruby_error(NULL, "proxy_hook must return a string or nil");
}
return EVENT_HOOK_STATUS_NEXT;
}
static enum evhook_status
script_hook_quit(va_list ap, void *data)
{
VALUE args[1];
int error;
erb_protected_method_call("quit_hook", 0, args, &error);
if (error)
erb_report_error(NULL, error);
return EVENT_HOOK_STATUS_NEXT;
}
struct event_hook_info ruby_scripting_hooks[] = {
{ "goto-url", 0, script_hook_goto_url, {NULL} },
{ "follow-url", 0, script_hook_follow_url, {NULL} },
{ "pre-format-html", 0, script_hook_pre_format_html, {NULL} },
{ "get-proxy", 0, script_hook_get_proxy, {NULL} },
{ "quit", 0, script_hook_quit, {NULL} },
NULL_EVENT_HOOK_INFO,
};