2016-03-19 15:14:35 +00:00
|
|
|
#include "first.h"
|
|
|
|
|
2019-10-27 19:19:01 +00:00
|
|
|
#include "plugin.h"
|
|
|
|
|
|
|
|
|
|
|
|
#if defined(HAVE_PCRE_H) /* do nothing if PCRE not available */
|
|
|
|
#if defined(HAVE_GDBM_H) || defined(USE_MEMCACHED) /* at least one required */
|
|
|
|
|
|
|
|
|
2005-07-07 22:48:42 +00:00
|
|
|
#include "base.h"
|
|
|
|
#include "log.h"
|
|
|
|
#include "buffer.h"
|
2018-09-09 05:50:33 +00:00
|
|
|
#include "http_header.h"
|
2005-07-07 22:48:42 +00:00
|
|
|
|
2018-03-25 07:45:05 +00:00
|
|
|
#include <sys/stat.h>
|
2009-10-11 14:31:42 +00:00
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
2005-07-07 22:48:42 +00:00
|
|
|
#if defined(HAVE_GDBM_H)
|
2019-10-27 19:19:01 +00:00
|
|
|
#include "fdevent.h"
|
2009-10-11 14:31:42 +00:00
|
|
|
# include <gdbm.h>
|
2005-07-07 22:48:42 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#if defined(HAVE_PCRE_H)
|
2009-10-11 14:31:42 +00:00
|
|
|
# include <pcre.h>
|
2005-07-07 22:48:42 +00:00
|
|
|
#endif
|
|
|
|
|
2016-01-03 14:48:11 +00:00
|
|
|
#if defined(USE_MEMCACHED)
|
|
|
|
# include <libmemcached/memcached.h>
|
2005-07-07 22:48:42 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
/**
|
|
|
|
* this is a trigger_b4_dl for a lighttpd plugin
|
2006-10-04 13:26:23 +00:00
|
|
|
*
|
2005-07-07 22:48:42 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
typedef struct {
|
2019-10-27 19:19:01 +00:00
|
|
|
const buffer *deny_url;
|
|
|
|
pcre *trigger_regex;
|
|
|
|
pcre *download_regex;
|
|
|
|
#if defined(HAVE_GDBM_H)
|
|
|
|
GDBM_FILE db;
|
|
|
|
#endif
|
|
|
|
#if defined(USE_MEMCACHED)
|
|
|
|
memcached_st *memc;
|
|
|
|
const buffer *mc_namespace;
|
|
|
|
#endif
|
|
|
|
unsigned short trigger_timeout;
|
|
|
|
unsigned short debug;
|
2005-07-07 22:48:42 +00:00
|
|
|
} plugin_config;
|
|
|
|
|
|
|
|
typedef struct {
|
2019-10-27 19:19:01 +00:00
|
|
|
PLUGIN_DATA;
|
|
|
|
plugin_config defaults;
|
|
|
|
plugin_config conf;
|
2005-07-07 22:48:42 +00:00
|
|
|
} plugin_data;
|
|
|
|
|
|
|
|
INIT_FUNC(mod_trigger_b4_dl_init) {
|
2019-10-27 19:19:01 +00:00
|
|
|
return calloc(1, sizeof(plugin_data));
|
2005-07-07 22:48:42 +00:00
|
|
|
}
|
|
|
|
|
2019-11-19 08:39:40 +00:00
|
|
|
FREE_FUNC(mod_trigger_b4_dl_free) {
|
|
|
|
plugin_data *p = p_d;
|
2019-10-27 19:19:01 +00:00
|
|
|
if (NULL == p->cvlist) return;
|
|
|
|
/* (init i to 0 if global context; to 1 to skip empty global context) */
|
|
|
|
for (int i = !p->cvlist[0].v.u2[1], used = p->nconfig; i < used; ++i) {
|
|
|
|
config_plugin_value_t *cpv = p->cvlist + p->cvlist[i].v.u2[0];
|
|
|
|
for (; -1 != cpv->k_id; ++cpv) {
|
|
|
|
if (cpv->vtype != T_CONFIG_LOCAL || NULL == cpv->v.v) continue;
|
|
|
|
switch (cpv->k_id) {
|
|
|
|
#if defined(HAVE_GDBM_H)
|
|
|
|
case 0: /* trigger-before-download.gdbm-filename */
|
|
|
|
gdbm_close(cpv->v.v);
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
case 1: /* trigger-before-download.trigger-url */
|
|
|
|
pcre_free(cpv->v.v);
|
|
|
|
break;
|
|
|
|
case 2: /* trigger-before-download.download-url */
|
|
|
|
pcre_free(cpv->v.v);
|
|
|
|
break;
|
|
|
|
#if defined(USE_MEMCACHED)
|
|
|
|
case 5: /* trigger-before-download.memcache-hosts */
|
|
|
|
memcached_free(cpv->v.v);
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2005-07-07 22:48:42 +00:00
|
|
|
}
|
|
|
|
|
2019-10-27 19:19:01 +00:00
|
|
|
static int mod_trigger_b4_dl_init_gdbm(server * const srv, config_plugin_value_t * const cpv) {
|
|
|
|
if (buffer_string_is_empty(cpv->v.b)) {
|
|
|
|
cpv->v.v = NULL;
|
|
|
|
return 1;
|
|
|
|
}
|
2005-07-07 22:48:42 +00:00
|
|
|
|
2019-10-27 19:19:01 +00:00
|
|
|
#if defined(HAVE_GDBM_H)
|
2006-10-04 13:26:23 +00:00
|
|
|
|
2019-10-27 19:19:01 +00:00
|
|
|
GDBM_FILE db = gdbm_open(cpv->v.b->ptr, 4096, GDBM_WRCREAT | GDBM_NOLOCK,
|
|
|
|
S_IRUSR | S_IWUSR, 0);
|
2006-10-04 13:26:23 +00:00
|
|
|
|
2019-10-27 19:19:01 +00:00
|
|
|
if (db) {
|
|
|
|
cpv->v.v = db;
|
|
|
|
cpv->vtype = T_CONFIG_LOCAL;
|
|
|
|
fdevent_setfd_cloexec(gdbm_fdesc(db));
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
log_error(srv->errh, __FILE__, __LINE__,
|
|
|
|
"gdbm-open failed %s", cpv->v.b->ptr);
|
|
|
|
return 0;
|
|
|
|
}
|
2006-10-04 13:26:23 +00:00
|
|
|
|
2019-10-27 19:19:01 +00:00
|
|
|
#else
|
2006-10-04 13:26:23 +00:00
|
|
|
|
2019-10-27 19:19:01 +00:00
|
|
|
UNUSED(srv);
|
|
|
|
return 1;
|
2006-10-04 13:26:23 +00:00
|
|
|
|
2019-10-27 19:19:01 +00:00
|
|
|
#endif
|
|
|
|
}
|
2016-01-03 14:48:11 +00:00
|
|
|
|
2019-10-27 19:19:01 +00:00
|
|
|
static int mod_trigger_b4_dl_init_memcached(server * const srv, config_plugin_value_t * const cpv) {
|
|
|
|
const array * const mc_hosts = cpv->v.a;
|
|
|
|
if (0 == mc_hosts->used) {
|
|
|
|
cpv->v.v = NULL;
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
#if defined(USE_MEMCACHED)
|
|
|
|
|
|
|
|
buffer * const opts = srv->tmp_buf;
|
|
|
|
buffer_clear(opts);
|
|
|
|
for (uint32_t k = 0; k < mc_hosts->used; ++k) {
|
|
|
|
const data_string * const ds = (const data_string *)mc_hosts->data[k];
|
|
|
|
buffer_append_string_len(opts, CONST_STR_LEN(" --SERVER="));
|
|
|
|
buffer_append_string_buffer(opts, &ds->value);
|
|
|
|
}
|
|
|
|
|
|
|
|
cpv->v.v = memcached(opts->ptr+1, buffer_string_length(opts)-1);
|
|
|
|
|
|
|
|
if (cpv->v.v) {
|
|
|
|
cpv->vtype = T_CONFIG_LOCAL;
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
log_error(srv->errh, __FILE__, __LINE__,
|
|
|
|
"configuring memcached failed for option string: %s", opts->ptr);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
#else
|
|
|
|
|
|
|
|
log_error(srv->errh, __FILE__, __LINE__,
|
|
|
|
"memcache support is not compiled in but "
|
|
|
|
"trigger-before-download.memcache-hosts is set; aborting");
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
#endif
|
|
|
|
}
|
2016-01-03 14:48:11 +00:00
|
|
|
|
2019-10-27 19:19:01 +00:00
|
|
|
static int mod_trigger_b4_dl_init_regex(server * const srv, config_plugin_value_t * const cpv, const char * const str) {
|
|
|
|
const buffer * const b = cpv->v.b;
|
|
|
|
if (buffer_string_is_empty(b)) {
|
|
|
|
cpv->v.v = NULL;
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
const char *errptr;
|
|
|
|
int erroff;
|
|
|
|
cpv->v.v = pcre_compile(b->ptr, 0, &errptr, &erroff, NULL);
|
|
|
|
|
|
|
|
if (cpv->v.v) {
|
|
|
|
cpv->vtype = T_CONFIG_LOCAL;
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
log_error(srv->errh, __FILE__, __LINE__,
|
|
|
|
"compiling regex for %s failed: %s pos: %d",
|
|
|
|
str, b->ptr, erroff);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
2006-10-04 13:26:23 +00:00
|
|
|
|
2019-10-27 19:19:01 +00:00
|
|
|
static void mod_trigger_b4_dl_merge_config_cpv(plugin_config * const pconf, const config_plugin_value_t * const cpv) {
|
|
|
|
switch (cpv->k_id) { /* index into static config_plugin_keys_t cpk[] */
|
|
|
|
case 0: /* trigger-before-download.gdbm-filename */
|
|
|
|
#if defined(HAVE_GDBM_H)
|
|
|
|
if (cpv->vtype != T_CONFIG_LOCAL) break;
|
|
|
|
pconf->db = cpv->v.v;
|
|
|
|
#endif
|
|
|
|
break;
|
|
|
|
case 1: /* trigger-before-download.trigger-url */
|
|
|
|
if (cpv->vtype != T_CONFIG_LOCAL) break;
|
|
|
|
pconf->trigger_regex = cpv->v.v;
|
|
|
|
break;
|
|
|
|
case 2: /* trigger-before-download.download-url */
|
|
|
|
if (cpv->vtype != T_CONFIG_LOCAL) break;
|
|
|
|
pconf->download_regex = cpv->v.v;
|
|
|
|
break;
|
|
|
|
case 3: /* trigger-before-download.deny-url */
|
|
|
|
pconf->deny_url = cpv->v.b;
|
|
|
|
break;
|
|
|
|
case 4: /* trigger-before-download.trigger-timeout */
|
|
|
|
pconf->trigger_timeout = cpv->v.shrt;
|
|
|
|
break;
|
|
|
|
case 5: /* trigger-before-download.memcache-hosts */
|
|
|
|
#if defined(USE_MEMCACHED)
|
|
|
|
if (cpv->vtype != T_CONFIG_LOCAL) break;
|
|
|
|
pconf->memc = cpv->v.v;
|
|
|
|
#endif
|
|
|
|
break;
|
|
|
|
case 6: /* trigger-before-download.memcache-namespace */
|
|
|
|
#if defined(USE_MEMCACHED)
|
|
|
|
pconf->mc_namespace = cpv->v.b;
|
|
|
|
#endif
|
|
|
|
break;
|
|
|
|
case 7: /* trigger-before-download.debug */
|
|
|
|
pconf->debug = cpv->v.u;
|
|
|
|
break;
|
|
|
|
default:/* should not happen */
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
2006-10-04 13:26:23 +00:00
|
|
|
|
2019-10-27 19:19:01 +00:00
|
|
|
static void mod_trigger_b4_dl_merge_config(plugin_config * const pconf, const config_plugin_value_t *cpv) {
|
|
|
|
do {
|
|
|
|
mod_trigger_b4_dl_merge_config_cpv(pconf, cpv);
|
|
|
|
} while ((++cpv)->k_id != -1);
|
|
|
|
}
|
2019-10-17 05:27:52 +00:00
|
|
|
|
2020-01-13 02:51:12 +00:00
|
|
|
static void mod_trigger_b4_dl_patch_config(request_st * const r, plugin_data * const p) {
|
2019-10-27 19:19:01 +00:00
|
|
|
memcpy(&p->conf, &p->defaults, sizeof(plugin_config));
|
|
|
|
for (int i = 1, used = p->nconfig; i < used; ++i) {
|
2020-01-13 02:51:12 +00:00
|
|
|
if (config_check_cond(r, (uint32_t)p->cvlist[i].k_id))
|
2019-10-27 19:19:01 +00:00
|
|
|
mod_trigger_b4_dl_merge_config(&p->conf,
|
|
|
|
p->cvlist + p->cvlist[i].v.u2[0]);
|
|
|
|
}
|
|
|
|
}
|
2006-10-04 13:26:23 +00:00
|
|
|
|
2019-10-27 19:19:01 +00:00
|
|
|
SETDEFAULTS_FUNC(mod_trigger_b4_dl_set_defaults) {
|
|
|
|
static const config_plugin_keys_t cpk[] = {
|
|
|
|
{ CONST_STR_LEN("trigger-before-download.gdbm-filename"),
|
|
|
|
T_CONFIG_STRING,
|
|
|
|
T_CONFIG_SCOPE_CONNECTION }
|
|
|
|
,{ CONST_STR_LEN("trigger-before-download.trigger-url"),
|
|
|
|
T_CONFIG_STRING,
|
|
|
|
T_CONFIG_SCOPE_CONNECTION }
|
|
|
|
,{ CONST_STR_LEN("trigger-before-download.download-url"),
|
|
|
|
T_CONFIG_STRING,
|
|
|
|
T_CONFIG_SCOPE_CONNECTION }
|
|
|
|
,{ CONST_STR_LEN("trigger-before-download.deny-url"),
|
|
|
|
T_CONFIG_STRING,
|
|
|
|
T_CONFIG_SCOPE_CONNECTION }
|
|
|
|
,{ CONST_STR_LEN("trigger-before-download.trigger-timeout"),
|
|
|
|
T_CONFIG_SHORT,
|
|
|
|
T_CONFIG_SCOPE_CONNECTION }
|
|
|
|
,{ CONST_STR_LEN("trigger-before-download.memcache-hosts"),
|
2019-12-08 00:15:55 +00:00
|
|
|
T_CONFIG_ARRAY_VLIST,
|
2019-10-27 19:19:01 +00:00
|
|
|
T_CONFIG_SCOPE_CONNECTION }
|
|
|
|
,{ CONST_STR_LEN("trigger-before-download.memcache-namespace"),
|
|
|
|
T_CONFIG_STRING,
|
|
|
|
T_CONFIG_SCOPE_CONNECTION }
|
|
|
|
,{ CONST_STR_LEN("trigger-before-download.debug"),
|
|
|
|
T_CONFIG_SHORT,
|
|
|
|
T_CONFIG_SCOPE_CONNECTION }
|
|
|
|
,{ NULL, 0,
|
|
|
|
T_CONFIG_UNSET,
|
|
|
|
T_CONFIG_SCOPE_UNSET }
|
|
|
|
};
|
|
|
|
|
|
|
|
plugin_data * const p = p_d;
|
|
|
|
if (!config_plugin_values_init(srv, p, cpk, "mod_trigger_b4_dl"))
|
|
|
|
return HANDLER_ERROR;
|
|
|
|
|
|
|
|
/* process and validate config directives
|
|
|
|
* (init i to 0 if global context; to 1 to skip empty global context) */
|
|
|
|
for (int i = !p->cvlist[0].v.u2[1]; i < p->nconfig; ++i) {
|
|
|
|
config_plugin_value_t *cpv = p->cvlist + p->cvlist[i].v.u2[0];
|
|
|
|
for (; -1 != cpv->k_id; ++cpv) {
|
|
|
|
switch (cpv->k_id) {
|
|
|
|
case 0: /* trigger-before-download.gdbm-filename */
|
|
|
|
if (!mod_trigger_b4_dl_init_gdbm(srv, cpv))
|
|
|
|
return HANDLER_ERROR;
|
|
|
|
break;
|
|
|
|
case 1: /* trigger-before-download.trigger-url */
|
|
|
|
if (!mod_trigger_b4_dl_init_regex(srv, cpv, "trigger-url"))
|
|
|
|
return HANDLER_ERROR;
|
|
|
|
break;
|
|
|
|
case 2: /* trigger-before-download.download-url */
|
|
|
|
if (!mod_trigger_b4_dl_init_regex(srv, cpv, "download-url"))
|
|
|
|
return HANDLER_ERROR;
|
|
|
|
break;
|
|
|
|
case 3: /* trigger-before-download.deny-url */
|
|
|
|
case 4: /* trigger-before-download.trigger-timeout */
|
|
|
|
break;
|
|
|
|
case 5: /* trigger-before-download.memcache-hosts */
|
|
|
|
if (!mod_trigger_b4_dl_init_memcached(srv, cpv))
|
|
|
|
return HANDLER_ERROR;
|
|
|
|
break;
|
|
|
|
case 6: /* trigger-before-download.memcache-namespace */
|
|
|
|
case 7: /* trigger-before-download.debug */
|
|
|
|
break;
|
|
|
|
default:/* should not happen */
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* initialize p->defaults from global config context */
|
|
|
|
if (p->nconfig > 0 && p->cvlist->v.u2[1]) {
|
|
|
|
const config_plugin_value_t *cpv = p->cvlist + p->cvlist->v.u2[0];
|
|
|
|
if (-1 != cpv->k_id)
|
|
|
|
mod_trigger_b4_dl_merge_config(&p->defaults, cpv);
|
|
|
|
}
|
|
|
|
|
|
|
|
return HANDLER_GO_ON;
|
|
|
|
}
|
2005-07-07 22:48:42 +00:00
|
|
|
|
2016-01-03 14:48:11 +00:00
|
|
|
#if defined(USE_MEMCACHED)
|
2019-10-27 19:19:01 +00:00
|
|
|
static void mod_trigger_b4_dl_memcached_key(buffer * const b, const plugin_data * const p, const buffer * const remote_ip) {
|
|
|
|
buffer_clear(b);
|
|
|
|
if (p->conf.mc_namespace)
|
|
|
|
buffer_copy_buffer(b, p->conf.mc_namespace);
|
|
|
|
buffer_append_string_buffer(b, remote_ip);
|
|
|
|
|
|
|
|
/* memcached can't handle spaces */
|
|
|
|
for (size_t i = 0, len = buffer_string_length(b); i < len; ++i) {
|
|
|
|
if (b->ptr[i] == ' ') b->ptr[i] = '-';
|
|
|
|
}
|
|
|
|
}
|
2005-07-07 22:48:42 +00:00
|
|
|
#endif
|
2006-10-04 13:26:23 +00:00
|
|
|
|
2020-01-13 02:51:12 +00:00
|
|
|
static handler_t mod_trigger_b4_dl_deny(request_st * const r, const plugin_data * const p) {
|
2019-10-27 19:19:01 +00:00
|
|
|
if (p->conf.deny_url) {
|
2020-01-13 02:51:12 +00:00
|
|
|
http_header_response_set(r, HTTP_HEADER_LOCATION,
|
2019-10-27 19:19:01 +00:00
|
|
|
CONST_STR_LEN("Location"),
|
|
|
|
CONST_BUF_LEN(p->conf.deny_url));
|
2020-01-13 02:51:12 +00:00
|
|
|
r->http_status = 307;
|
2019-10-27 19:19:01 +00:00
|
|
|
}
|
|
|
|
else {
|
2020-01-13 02:51:12 +00:00
|
|
|
log_error(r->conf.errh, __FILE__, __LINE__,
|
2019-10-27 19:19:01 +00:00
|
|
|
"trigger-before-download.deny-url not configured");
|
2020-01-13 02:51:12 +00:00
|
|
|
r->http_status = 500;
|
2019-10-27 19:19:01 +00:00
|
|
|
}
|
2020-01-13 02:51:12 +00:00
|
|
|
r->resp_body_finished = 1;
|
2019-10-27 19:19:01 +00:00
|
|
|
return HANDLER_FINISHED;
|
2005-07-07 22:48:42 +00:00
|
|
|
}
|
2018-09-26 00:49:25 +00:00
|
|
|
|
2005-07-07 22:48:42 +00:00
|
|
|
URIHANDLER_FUNC(mod_trigger_b4_dl_uri_handler) {
|
|
|
|
plugin_data *p = p_d;
|
|
|
|
|
|
|
|
int n;
|
|
|
|
# define N 10
|
|
|
|
int ovec[N * 3];
|
2006-10-04 13:26:23 +00:00
|
|
|
|
2020-01-13 02:51:12 +00:00
|
|
|
if (NULL != r->handler_module) return HANDLER_GO_ON;
|
2008-08-01 16:13:34 +00:00
|
|
|
|
2020-01-13 02:51:12 +00:00
|
|
|
mod_trigger_b4_dl_patch_config(r, p);
|
2006-10-04 13:26:23 +00:00
|
|
|
|
2005-07-09 20:18:10 +00:00
|
|
|
if (!p->conf.trigger_regex || !p->conf.download_regex) return HANDLER_GO_ON;
|
2006-10-04 13:26:23 +00:00
|
|
|
|
2016-01-03 14:48:11 +00:00
|
|
|
# if !defined(HAVE_GDBM_H) && !defined(USE_MEMCACHED)
|
2005-07-09 20:18:10 +00:00
|
|
|
return HANDLER_GO_ON;
|
2016-01-03 14:48:11 +00:00
|
|
|
# elif defined(HAVE_GDBM_H) && defined(USE_MEMCACHED)
|
|
|
|
if (!p->conf.db && !p->conf.memc) return HANDLER_GO_ON;
|
|
|
|
if (p->conf.db && p->conf.memc) {
|
2005-07-09 20:18:10 +00:00
|
|
|
/* can't decide which one */
|
2006-10-04 13:26:23 +00:00
|
|
|
|
2005-07-09 20:18:10 +00:00
|
|
|
return HANDLER_GO_ON;
|
|
|
|
}
|
|
|
|
# elif defined(HAVE_GDBM_H)
|
2005-07-10 06:25:10 +00:00
|
|
|
if (!p->conf.db) return HANDLER_GO_ON;
|
2005-07-09 20:18:10 +00:00
|
|
|
# else
|
2016-01-03 14:48:11 +00:00
|
|
|
if (!p->conf.memc) return HANDLER_GO_ON;
|
2005-07-09 20:18:10 +00:00
|
|
|
# endif
|
2006-10-04 13:26:23 +00:00
|
|
|
|
2019-10-27 19:19:01 +00:00
|
|
|
/* X-Forwarded-For contains the ip behind the proxy */
|
|
|
|
const buffer *remote_ip =
|
2020-01-13 02:51:12 +00:00
|
|
|
http_header_request_get(r, HTTP_HEADER_X_FORWARDED_FOR,
|
2019-10-27 19:19:01 +00:00
|
|
|
CONST_STR_LEN("X-Forwarded-For"));
|
|
|
|
if (NULL == remote_ip) {
|
2020-01-13 02:51:12 +00:00
|
|
|
remote_ip = r->con->dst_addr_buf;
|
2005-07-07 22:48:42 +00:00
|
|
|
}
|
2005-08-17 17:36:03 +00:00
|
|
|
|
|
|
|
if (p->conf.debug) {
|
2020-01-13 02:51:12 +00:00
|
|
|
log_error(r->conf.errh, __FILE__, __LINE__, "(debug) remote-ip: %s", remote_ip->ptr);
|
2005-08-17 17:36:03 +00:00
|
|
|
}
|
2006-10-04 13:26:23 +00:00
|
|
|
|
2019-12-04 06:35:27 +00:00
|
|
|
const time_t cur_ts = log_epoch_secs;
|
2019-12-04 06:35:27 +00:00
|
|
|
|
2005-07-07 22:48:42 +00:00
|
|
|
/* check if URL is a trigger -> insert IP into DB */
|
2020-01-13 02:51:12 +00:00
|
|
|
if ((n = pcre_exec(p->conf.trigger_regex, NULL, CONST_BUF_LEN(&r->uri.path), 0, 0, ovec, 3 * N)) < 0) {
|
2005-07-07 22:48:42 +00:00
|
|
|
if (n != PCRE_ERROR_NOMATCH) {
|
2020-01-13 02:51:12 +00:00
|
|
|
log_error(r->conf.errh, __FILE__, __LINE__,
|
2019-11-25 06:54:08 +00:00
|
|
|
"execution error while matching: %d", n);
|
2006-10-04 13:26:23 +00:00
|
|
|
|
2005-07-07 22:48:42 +00:00
|
|
|
return HANDLER_ERROR;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
# if defined(HAVE_GDBM_H)
|
2005-07-09 20:18:10 +00:00
|
|
|
if (p->conf.db) {
|
2005-07-07 22:48:42 +00:00
|
|
|
/* the trigger matched */
|
|
|
|
datum key, val;
|
2006-10-04 13:26:23 +00:00
|
|
|
|
2019-10-27 19:19:01 +00:00
|
|
|
*(const char **)&key.dptr = remote_ip->ptr;
|
|
|
|
key.dsize = buffer_string_length(remote_ip);
|
2006-10-04 13:26:23 +00:00
|
|
|
|
2019-12-04 06:35:27 +00:00
|
|
|
val.dptr = (char *)&cur_ts;
|
|
|
|
val.dsize = sizeof(cur_ts);
|
2006-10-04 13:26:23 +00:00
|
|
|
|
2005-07-07 22:48:42 +00:00
|
|
|
if (0 != gdbm_store(p->conf.db, key, val, GDBM_REPLACE)) {
|
2020-01-13 02:51:12 +00:00
|
|
|
log_error(r->conf.errh, __FILE__, __LINE__, "insert failed");
|
2005-07-07 22:48:42 +00:00
|
|
|
}
|
2005-07-09 20:18:10 +00:00
|
|
|
}
|
2005-07-07 22:48:42 +00:00
|
|
|
# endif
|
2016-01-03 14:48:11 +00:00
|
|
|
# if defined(USE_MEMCACHED)
|
|
|
|
if (p->conf.memc) {
|
2020-01-13 02:51:12 +00:00
|
|
|
buffer * const b = r->tmp_buf;
|
2019-10-27 19:19:01 +00:00
|
|
|
mod_trigger_b4_dl_memcached_key(b, p, remote_ip);
|
2006-10-04 13:26:23 +00:00
|
|
|
|
2005-08-17 17:36:03 +00:00
|
|
|
if (p->conf.debug) {
|
2020-01-13 02:51:12 +00:00
|
|
|
log_error(r->conf.errh, __FILE__, __LINE__, "(debug) triggered IP: %s", b->ptr);
|
2005-08-17 17:36:03 +00:00
|
|
|
}
|
|
|
|
|
2016-01-03 14:48:11 +00:00
|
|
|
if (MEMCACHED_SUCCESS != memcached_set(p->conf.memc,
|
2019-10-27 19:19:01 +00:00
|
|
|
CONST_BUF_LEN(b),
|
2019-12-04 06:35:27 +00:00
|
|
|
(const char *)&cur_ts, sizeof(cur_ts),
|
2005-07-07 22:48:42 +00:00
|
|
|
p->conf.trigger_timeout, 0)) {
|
2020-01-13 02:51:12 +00:00
|
|
|
log_error(r->conf.errh, __FILE__, __LINE__, "insert failed");
|
2005-07-07 22:48:42 +00:00
|
|
|
}
|
|
|
|
}
|
2005-07-09 20:18:10 +00:00
|
|
|
# endif
|
2005-07-07 22:48:42 +00:00
|
|
|
}
|
2006-10-04 13:26:23 +00:00
|
|
|
|
2005-07-07 22:48:42 +00:00
|
|
|
/* check if URL is a download -> check IP in DB, update timestamp */
|
2020-01-13 02:51:12 +00:00
|
|
|
if ((n = pcre_exec(p->conf.download_regex, NULL, CONST_BUF_LEN(&r->uri.path), 0, 0, ovec, 3 * N)) < 0) {
|
2005-07-07 22:48:42 +00:00
|
|
|
if (n != PCRE_ERROR_NOMATCH) {
|
2020-01-13 02:51:12 +00:00
|
|
|
log_error(r->conf.errh, __FILE__, __LINE__,
|
2019-11-25 06:54:08 +00:00
|
|
|
"execution error while matching: %d", n);
|
2005-07-07 22:48:42 +00:00
|
|
|
return HANDLER_ERROR;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
/* the download uri matched */
|
2006-10-04 13:26:23 +00:00
|
|
|
# if defined(HAVE_GDBM_H)
|
2005-07-07 22:48:42 +00:00
|
|
|
if (p->conf.db) {
|
|
|
|
datum key, val;
|
2005-07-09 20:18:10 +00:00
|
|
|
time_t last_hit;
|
2006-10-04 13:26:23 +00:00
|
|
|
|
2019-10-27 19:19:01 +00:00
|
|
|
*(const char **)&key.dptr = remote_ip->ptr;
|
|
|
|
key.dsize = buffer_string_length(remote_ip);
|
2006-10-04 13:26:23 +00:00
|
|
|
|
2005-07-07 22:48:42 +00:00
|
|
|
val = gdbm_fetch(p->conf.db, key);
|
2006-10-04 13:26:23 +00:00
|
|
|
|
2005-07-07 22:48:42 +00:00
|
|
|
if (val.dptr == NULL) {
|
|
|
|
/* not found, redirect */
|
2020-01-13 02:51:12 +00:00
|
|
|
return mod_trigger_b4_dl_deny(r, p);
|
2005-07-07 22:48:42 +00:00
|
|
|
}
|
2006-10-04 13:26:23 +00:00
|
|
|
|
2013-05-15 10:31:09 +00:00
|
|
|
memcpy(&last_hit, val.dptr, sizeof(time_t));
|
2006-10-04 13:26:23 +00:00
|
|
|
|
2005-07-07 22:48:42 +00:00
|
|
|
free(val.dptr);
|
2006-10-04 13:26:23 +00:00
|
|
|
|
2019-12-04 06:35:27 +00:00
|
|
|
if (cur_ts - last_hit > p->conf.trigger_timeout) {
|
2005-07-07 22:48:42 +00:00
|
|
|
/* found, but timeout, redirect */
|
2006-10-04 13:26:23 +00:00
|
|
|
|
2005-07-07 22:48:42 +00:00
|
|
|
if (p->conf.db) {
|
|
|
|
if (0 != gdbm_delete(p->conf.db, key)) {
|
2020-01-13 02:51:12 +00:00
|
|
|
log_error(r->conf.errh, __FILE__, __LINE__, "delete failed");
|
2005-07-07 22:48:42 +00:00
|
|
|
}
|
|
|
|
}
|
2006-10-04 13:26:23 +00:00
|
|
|
|
2020-01-13 02:51:12 +00:00
|
|
|
return mod_trigger_b4_dl_deny(r, p);
|
2005-07-07 22:48:42 +00:00
|
|
|
}
|
2006-10-04 13:26:23 +00:00
|
|
|
|
2019-12-04 06:35:27 +00:00
|
|
|
val.dptr = (char *)&cur_ts;
|
|
|
|
val.dsize = sizeof(cur_ts);
|
2006-10-04 13:26:23 +00:00
|
|
|
|
2005-07-07 22:48:42 +00:00
|
|
|
if (0 != gdbm_store(p->conf.db, key, val, GDBM_REPLACE)) {
|
2020-01-13 02:51:12 +00:00
|
|
|
log_error(r->conf.errh, __FILE__, __LINE__, "insert failed");
|
2005-07-07 22:48:42 +00:00
|
|
|
}
|
2005-07-09 20:18:10 +00:00
|
|
|
}
|
2005-07-07 22:48:42 +00:00
|
|
|
# endif
|
2016-01-03 14:48:11 +00:00
|
|
|
# if defined(USE_MEMCACHED)
|
|
|
|
if (p->conf.memc) {
|
2020-01-13 02:51:12 +00:00
|
|
|
buffer * const b = r->tmp_buf;
|
2019-10-27 19:19:01 +00:00
|
|
|
mod_trigger_b4_dl_memcached_key(b, p, remote_ip);
|
2006-10-04 13:26:23 +00:00
|
|
|
|
2005-08-17 17:36:03 +00:00
|
|
|
if (p->conf.debug) {
|
2020-01-13 02:51:12 +00:00
|
|
|
log_error(r->conf.errh, __FILE__, __LINE__, "(debug) checking IP: %s", b->ptr);
|
2005-08-17 17:36:03 +00:00
|
|
|
}
|
|
|
|
|
2005-07-07 22:48:42 +00:00
|
|
|
/**
|
2006-10-04 13:26:23 +00:00
|
|
|
*
|
2005-07-07 22:48:42 +00:00
|
|
|
* memcached is do expiration for us, as long as we can fetch it every thing is ok
|
2006-10-04 13:26:23 +00:00
|
|
|
* and the timestamp is updated
|
|
|
|
*
|
2005-07-07 22:48:42 +00:00
|
|
|
*/
|
2019-10-27 19:19:01 +00:00
|
|
|
if (MEMCACHED_SUCCESS != memcached_exist(p->conf.memc, CONST_BUF_LEN(b))) {
|
2020-01-13 02:51:12 +00:00
|
|
|
return mod_trigger_b4_dl_deny(r, p);
|
2005-07-07 22:48:42 +00:00
|
|
|
}
|
2006-10-04 13:26:23 +00:00
|
|
|
|
2005-07-07 22:48:42 +00:00
|
|
|
/* set a new timeout */
|
2016-01-03 14:48:11 +00:00
|
|
|
if (MEMCACHED_SUCCESS != memcached_set(p->conf.memc,
|
2019-10-27 19:19:01 +00:00
|
|
|
CONST_BUF_LEN(b),
|
2019-12-04 06:35:27 +00:00
|
|
|
(const char *)&cur_ts, sizeof(cur_ts),
|
2005-07-07 22:48:42 +00:00
|
|
|
p->conf.trigger_timeout, 0)) {
|
2020-01-13 02:51:12 +00:00
|
|
|
log_error(r->conf.errh, __FILE__, __LINE__, "insert failed");
|
2005-07-07 22:48:42 +00:00
|
|
|
}
|
|
|
|
}
|
2005-07-09 20:18:10 +00:00
|
|
|
# endif
|
2005-07-07 22:48:42 +00:00
|
|
|
}
|
2006-10-04 13:26:23 +00:00
|
|
|
|
2005-07-07 22:48:42 +00:00
|
|
|
return HANDLER_GO_ON;
|
|
|
|
}
|
|
|
|
|
|
|
|
#if defined(HAVE_GDBM_H)
|
2019-10-27 19:19:01 +00:00
|
|
|
static void mod_trigger_b4_dl_trigger_gdbm(GDBM_FILE db, const time_t cur_ts, const int trigger_timeout) {
|
2005-07-07 22:48:42 +00:00
|
|
|
datum key, val, okey;
|
|
|
|
okey.dptr = NULL;
|
2006-10-04 13:26:23 +00:00
|
|
|
|
|
|
|
/* according to the manual this loop + delete does delete all entries on its way
|
|
|
|
*
|
2005-07-07 22:48:42 +00:00
|
|
|
* we don't care as the next round will remove them. We don't have to perfect here.
|
|
|
|
*/
|
2019-10-27 19:19:01 +00:00
|
|
|
for (key = gdbm_firstkey(db); key.dptr; key = gdbm_nextkey(db, okey)) {
|
2005-07-07 22:48:42 +00:00
|
|
|
time_t last_hit;
|
|
|
|
if (okey.dptr) {
|
|
|
|
free(okey.dptr);
|
|
|
|
okey.dptr = NULL;
|
|
|
|
}
|
2006-10-04 13:26:23 +00:00
|
|
|
|
2019-10-27 19:19:01 +00:00
|
|
|
val = gdbm_fetch(db, key);
|
2006-10-04 13:26:23 +00:00
|
|
|
|
2013-05-15 10:31:09 +00:00
|
|
|
memcpy(&last_hit, val.dptr, sizeof(time_t));
|
2006-10-04 13:26:23 +00:00
|
|
|
|
2005-07-07 22:48:42 +00:00
|
|
|
free(val.dptr);
|
2006-10-04 13:26:23 +00:00
|
|
|
|
2019-10-27 19:19:01 +00:00
|
|
|
if (cur_ts - last_hit > trigger_timeout) {
|
|
|
|
gdbm_delete(db, key);
|
2005-07-07 22:48:42 +00:00
|
|
|
}
|
2006-10-04 13:26:23 +00:00
|
|
|
|
2005-07-07 22:48:42 +00:00
|
|
|
okey = key;
|
|
|
|
}
|
|
|
|
if (okey.dptr) free(okey.dptr);
|
2006-10-04 13:26:23 +00:00
|
|
|
|
2005-07-07 22:48:42 +00:00
|
|
|
/* reorg once a day */
|
2019-10-27 19:19:01 +00:00
|
|
|
if ((cur_ts % (60 * 60 * 24) != 0)) gdbm_reorganize(db);
|
|
|
|
}
|
|
|
|
|
|
|
|
TRIGGER_FUNC(mod_trigger_b4_dl_handle_trigger) {
|
|
|
|
/* check DB each minute */
|
2019-12-04 06:35:27 +00:00
|
|
|
const time_t cur_ts = log_epoch_secs;
|
2019-10-27 19:19:01 +00:00
|
|
|
if (cur_ts % 60 != 0) return HANDLER_GO_ON;
|
2019-12-04 06:35:27 +00:00
|
|
|
UNUSED(srv);
|
2019-10-27 19:19:01 +00:00
|
|
|
|
|
|
|
plugin_data * const p = p_d;
|
|
|
|
|
|
|
|
/* (init i to 0 if global context; to 1 to skip empty global context) */
|
|
|
|
for (int i = !p->cvlist[0].v.u2[1]; i < p->nconfig; ++i) {
|
|
|
|
config_plugin_value_t *cpv = p->cvlist + p->cvlist[i].v.u2[0];
|
|
|
|
void *db = NULL;
|
|
|
|
int timeout = (int)p->defaults.trigger_timeout;
|
|
|
|
for (; -1 != cpv->k_id; ++cpv) {
|
|
|
|
switch (cpv->k_id) {
|
|
|
|
case 0: /* trigger-before-download.gdbm-filename */
|
|
|
|
if (cpv->vtype == T_CONFIG_LOCAL && NULL != cpv->v.v)
|
|
|
|
db = cpv->v.v;
|
|
|
|
break;
|
|
|
|
case 4: /* trigger-before-download.trigger-timeout */
|
|
|
|
timeout = (int)cpv->v.shrt;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (db)
|
|
|
|
mod_trigger_b4_dl_trigger_gdbm(db, cur_ts, timeout);
|
|
|
|
}
|
|
|
|
|
|
|
|
return HANDLER_GO_ON;
|
2005-07-07 22:48:42 +00:00
|
|
|
}
|
2005-07-15 17:45:57 +00:00
|
|
|
#endif
|
2005-07-07 22:48:42 +00:00
|
|
|
|
2019-10-27 19:19:01 +00:00
|
|
|
|
|
|
|
#endif /* defined(HAVE_PCRE_H) */
|
|
|
|
#endif /* defined(HAVE_GDBM_H) || defined(USE_MEMCACHED) */
|
|
|
|
|
2005-07-07 22:48:42 +00:00
|
|
|
|
2009-03-07 21:05:37 +00:00
|
|
|
int mod_trigger_b4_dl_plugin_init(plugin *p);
|
2005-07-07 22:48:42 +00:00
|
|
|
int mod_trigger_b4_dl_plugin_init(plugin *p) {
|
|
|
|
p->version = LIGHTTPD_VERSION_ID;
|
2019-10-19 04:30:54 +00:00
|
|
|
p->name = "trigger_b4_dl";
|
2006-10-04 13:26:23 +00:00
|
|
|
|
2019-10-27 19:19:01 +00:00
|
|
|
#if defined(HAVE_PCRE_H) /* do nothing if PCRE not available */
|
|
|
|
#if defined(HAVE_GDBM_H) || defined(USE_MEMCACHED) /* at least one required */
|
|
|
|
|
2005-07-07 22:48:42 +00:00
|
|
|
p->init = mod_trigger_b4_dl_init;
|
|
|
|
p->handle_uri_clean = mod_trigger_b4_dl_uri_handler;
|
|
|
|
p->set_defaults = mod_trigger_b4_dl_set_defaults;
|
2005-07-15 17:45:57 +00:00
|
|
|
#if defined(HAVE_GDBM_H)
|
2005-07-07 22:48:42 +00:00
|
|
|
p->handle_trigger = mod_trigger_b4_dl_handle_trigger;
|
2005-07-15 17:45:57 +00:00
|
|
|
#endif
|
2005-07-07 22:48:42 +00:00
|
|
|
p->cleanup = mod_trigger_b4_dl_free;
|
2006-10-04 13:26:23 +00:00
|
|
|
|
2019-10-27 19:19:01 +00:00
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
|
2005-07-07 22:48:42 +00:00
|
|
|
return 0;
|
|
|
|
}
|