mirror of
https://github.com/civetweb/civetweb
synced 2025-03-28 21:13:27 +00:00
Allow filtering and formating the access log in Lua
The lua background state can export a log() function. This function can filter the log by returning true (log) or false (do not log), or it can format a log message before it is written to the log file by returning it as string.
This commit is contained in:
parent
52425fc70f
commit
e9209760a3
@ -1090,7 +1090,7 @@ function instead.
|
||||
A Lua background script may define the following functions:
|
||||
start() -- called wnen the server is started
|
||||
stop() -- called when the server is stopped
|
||||
|
||||
log() -- called when an access log entry is created
|
||||
|
||||
See example Lua script :
|
||||
[background.lua](https://github.com/civetweb/civetweb/blob/master/test/lua_backbround_script_timer.lua).
|
||||
|
163
src/civetweb.c
163
src/civetweb.c
@ -2535,8 +2535,9 @@ struct mg_context {
|
||||
|
||||
/* Lua specific: Background operations and shared websockets */
|
||||
#if defined(USE_LUA)
|
||||
void *lua_background_state;
|
||||
void *lua_background_state; /* lua_State (here as void *) */
|
||||
pthread_mutex_t lua_bg_mutex; /* Protect background state */
|
||||
int lua_bg_log_available; /* Use Lua background state for access log */
|
||||
#endif
|
||||
|
||||
/* Server nonce */
|
||||
@ -15271,12 +15272,57 @@ log_access(const struct mg_connection *conn)
|
||||
const char *referer;
|
||||
const char *user_agent;
|
||||
|
||||
char buf[4096];
|
||||
char log_buf[4096];
|
||||
|
||||
if (!conn || !conn->dom_ctx) {
|
||||
return;
|
||||
}
|
||||
|
||||
/* Set log message to "empty" */
|
||||
log_buf[0] = 0;
|
||||
|
||||
#if defined(USE_LUA)
|
||||
if (conn->phys_ctx->lua_bg_log_available) {
|
||||
int ret;
|
||||
struct mg_context *ctx = conn->phys_ctx;
|
||||
lua_State *lstate = (lua_State *)ctx->lua_background_state;
|
||||
pthread_mutex_lock(&ctx->lua_bg_mutex);
|
||||
/* call "log()" in Lua */
|
||||
lua_getglobal(lstate, "log");
|
||||
prepare_lua_request_info_inner(conn, lstate);
|
||||
|
||||
ret = lua_pcall(lstate, /* args */ 1, /* results */ 1, 0);
|
||||
if (ret == 0) {
|
||||
int t = lua_type(lstate, -1);
|
||||
if (t == LUA_TBOOLEAN) {
|
||||
if (lua_toboolean(lstate, -1) == 0) {
|
||||
/* log() returned false: do not log */
|
||||
pthread_mutex_unlock(&ctx->lua_bg_mutex);
|
||||
return;
|
||||
}
|
||||
/* log returned true: continue logging */
|
||||
} else if (t == LUA_TSTRING) {
|
||||
size_t len;
|
||||
const char *txt = lua_tolstring(lstate, -1, &len);
|
||||
if ((len == 0) || (*txt == 0)) {
|
||||
/* log() returned empty string: do not log */
|
||||
pthread_mutex_unlock(&ctx->lua_bg_mutex);
|
||||
return;
|
||||
}
|
||||
/* Copy test from Lua into log_buf */
|
||||
if (len >= sizeof(log_buf)) {
|
||||
len = sizeof(log_buf) - 1;
|
||||
}
|
||||
memcpy(log_buf, txt, len);
|
||||
log_buf[len] = 0;
|
||||
}
|
||||
} else {
|
||||
lua_cry(conn, ret, lstate, "lua_background_script", "log");
|
||||
}
|
||||
pthread_mutex_unlock(&ctx->lua_bg_mutex);
|
||||
}
|
||||
#endif
|
||||
|
||||
if (conn->dom_ctx->config[ACCESS_LOG_FILE] != NULL) {
|
||||
if (mg_fopen(conn,
|
||||
conn->dom_ctx->config[ACCESS_LOG_FILE],
|
||||
@ -15296,40 +15342,45 @@ log_access(const struct mg_connection *conn)
|
||||
return;
|
||||
}
|
||||
|
||||
tm = localtime(&conn->conn_birth_time);
|
||||
if (tm != NULL) {
|
||||
strftime(date, sizeof(date), "%d/%b/%Y:%H:%M:%S %z", tm);
|
||||
} else {
|
||||
mg_strlcpy(date, "01/Jan/1970:00:00:00 +0000", sizeof(date));
|
||||
date[sizeof(date) - 1] = '\0';
|
||||
/* If we did not get a log message from Lua, create it here. */
|
||||
if (!log_buf[0]) {
|
||||
tm = localtime(&conn->conn_birth_time);
|
||||
if (tm != NULL) {
|
||||
strftime(date, sizeof(date), "%d/%b/%Y:%H:%M:%S %z", tm);
|
||||
} else {
|
||||
mg_strlcpy(date, "01/Jan/1970:00:00:00 +0000", sizeof(date));
|
||||
date[sizeof(date) - 1] = '\0';
|
||||
}
|
||||
|
||||
ri = &conn->request_info;
|
||||
|
||||
sockaddr_to_string(src_addr, sizeof(src_addr), &conn->client.rsa);
|
||||
referer = header_val(conn, "Referer");
|
||||
user_agent = header_val(conn, "User-Agent");
|
||||
|
||||
mg_snprintf(conn,
|
||||
NULL, /* Ignore truncation in access log */
|
||||
log_buf,
|
||||
sizeof(log_buf),
|
||||
"%s - %s [%s] \"%s %s%s%s HTTP/%s\" %d %" INT64_FMT
|
||||
" %s %s",
|
||||
src_addr,
|
||||
(ri->remote_user == NULL) ? "-" : ri->remote_user,
|
||||
date,
|
||||
ri->request_method ? ri->request_method : "-",
|
||||
ri->request_uri ? ri->request_uri : "-",
|
||||
ri->query_string ? "?" : "",
|
||||
ri->query_string ? ri->query_string : "",
|
||||
ri->http_version,
|
||||
conn->status_code,
|
||||
conn->num_bytes_sent,
|
||||
referer,
|
||||
user_agent);
|
||||
}
|
||||
|
||||
ri = &conn->request_info;
|
||||
|
||||
sockaddr_to_string(src_addr, sizeof(src_addr), &conn->client.rsa);
|
||||
referer = header_val(conn, "Referer");
|
||||
user_agent = header_val(conn, "User-Agent");
|
||||
|
||||
mg_snprintf(conn,
|
||||
NULL, /* Ignore truncation in access log */
|
||||
buf,
|
||||
sizeof(buf),
|
||||
"%s - %s [%s] \"%s %s%s%s HTTP/%s\" %d %" INT64_FMT " %s %s",
|
||||
src_addr,
|
||||
(ri->remote_user == NULL) ? "-" : ri->remote_user,
|
||||
date,
|
||||
ri->request_method ? ri->request_method : "-",
|
||||
ri->request_uri ? ri->request_uri : "-",
|
||||
ri->query_string ? "?" : "",
|
||||
ri->query_string ? ri->query_string : "",
|
||||
ri->http_version,
|
||||
conn->status_code,
|
||||
conn->num_bytes_sent,
|
||||
referer,
|
||||
user_agent);
|
||||
|
||||
/* Here we have a log message in log_buf. Call the callback */
|
||||
if (conn->phys_ctx->callbacks.log_access) {
|
||||
if (conn->phys_ctx->callbacks.log_access(conn, buf)) {
|
||||
if (conn->phys_ctx->callbacks.log_access(conn, log_buf)) {
|
||||
/* do not log if callack returns non-zero */
|
||||
if (fi.access.fp) {
|
||||
mg_fclose(&fi.access);
|
||||
@ -15338,10 +15389,11 @@ log_access(const struct mg_connection *conn)
|
||||
}
|
||||
}
|
||||
|
||||
/* Store in file */
|
||||
if (fi.access.fp) {
|
||||
int ok = 1;
|
||||
flockfile(fi.access.fp);
|
||||
if (fprintf(fi.access.fp, "%s\n", buf) < 1) {
|
||||
if (fprintf(fi.access.fp, "%s\n", log_buf) < 1) {
|
||||
ok = 0;
|
||||
}
|
||||
if (fflush(fi.access.fp) != 0) {
|
||||
@ -15359,7 +15411,7 @@ log_access(const struct mg_connection *conn)
|
||||
}
|
||||
}
|
||||
#else
|
||||
#error Must either enable filesystems or provide a custom log_access implementation
|
||||
#error "Either enable filesystems or provide a custom log_access implementation"
|
||||
#endif /* Externally provided function */
|
||||
|
||||
|
||||
@ -19178,9 +19230,30 @@ master_thread_run(struct mg_context *ctx)
|
||||
if (ctx->lua_background_state) {
|
||||
lua_State *lstate = (lua_State *)ctx->lua_background_state;
|
||||
pthread_mutex_lock(&ctx->lua_bg_mutex);
|
||||
|
||||
/* call "start()" in Lua */
|
||||
lua_getglobal(lstate, "start");
|
||||
(void)lua_pcall(lstate, /* args */ 0, /* results */ 0, 0);
|
||||
if (lua_type(lstate, -1) == LUA_TFUNCTION) {
|
||||
int ret = lua_pcall(lstate, /* args */ 0, /* results */ 0, 0);
|
||||
if (ret != 0) {
|
||||
struct mg_connection fc;
|
||||
lua_cry(fake_connection(&fc, ctx),
|
||||
ret,
|
||||
lstate,
|
||||
"lua_background_script",
|
||||
"start");
|
||||
}
|
||||
} else {
|
||||
lua_pop(lstate, 1);
|
||||
}
|
||||
|
||||
/* determine if there is a "log()" function in Lua background script */
|
||||
lua_getglobal(lstate, "log");
|
||||
if (lua_type(lstate, -1) == LUA_TFUNCTION) {
|
||||
ctx->lua_bg_log_available = 1;
|
||||
}
|
||||
lua_pop(lstate, 1);
|
||||
|
||||
pthread_mutex_unlock(&ctx->lua_bg_mutex);
|
||||
}
|
||||
#endif
|
||||
@ -19240,11 +19313,24 @@ master_thread_run(struct mg_context *ctx)
|
||||
/* Free Lua state of lua background task */
|
||||
if (ctx->lua_background_state) {
|
||||
lua_State *lstate = (lua_State *)ctx->lua_background_state;
|
||||
ctx->lua_bg_log_available = 0;
|
||||
|
||||
/* call "stop()" in Lua */
|
||||
pthread_mutex_lock(&ctx->lua_bg_mutex);
|
||||
lua_getglobal(lstate, "stop");
|
||||
(void)lua_pcall(lstate, /* args */ 0, /* results */ 0, 0);
|
||||
if (lua_type(lstate, -1) == LUA_TFUNCTION) {
|
||||
int ret = lua_pcall(lstate, /* args */ 0, /* results */ 0, 0);
|
||||
if (ret != 0) {
|
||||
struct mg_connection fc;
|
||||
lua_cry(fake_connection(&fc, ctx),
|
||||
ret,
|
||||
lstate,
|
||||
"lua_background_script",
|
||||
"stop");
|
||||
}
|
||||
}
|
||||
lua_close(lstate);
|
||||
|
||||
ctx->lua_background_state = 0;
|
||||
pthread_mutex_unlock(&ctx->lua_bg_mutex);
|
||||
}
|
||||
@ -19784,6 +19870,7 @@ static
|
||||
|
||||
#if defined(USE_LUA)
|
||||
/* If a Lua background script has been configured, start it. */
|
||||
ctx->lua_bg_log_available = 0;
|
||||
if (ctx->dd.config[LUA_BACKGROUND_SCRIPT] != NULL) {
|
||||
char ebuf[256];
|
||||
struct vec opt_vec;
|
||||
@ -19806,7 +19893,7 @@ static
|
||||
error->text,
|
||||
error->text_buffer_size,
|
||||
"Error in script %s: %s",
|
||||
config_options[DOCUMENT_ROOT].name,
|
||||
config_options[LUA_BACKGROUND_SCRIPT].name,
|
||||
ebuf);
|
||||
}
|
||||
pthread_mutex_unlock(&ctx->lua_bg_mutex);
|
||||
|
@ -156,7 +156,7 @@ reg_function(struct lua_State *L, const char *name, lua_CFunction func)
|
||||
|
||||
|
||||
static void
|
||||
lua_cry(struct mg_connection *conn,
|
||||
lua_cry(const struct mg_connection *conn,
|
||||
int err,
|
||||
lua_State *L,
|
||||
const char *lua_title,
|
||||
@ -2458,13 +2458,11 @@ enum {
|
||||
|
||||
|
||||
static void
|
||||
prepare_lua_request_info(struct mg_connection *conn, lua_State *L)
|
||||
prepare_lua_request_info_inner(const struct mg_connection *conn, lua_State *L)
|
||||
{
|
||||
const char *s;
|
||||
int i;
|
||||
|
||||
/* Export mg.request_info */
|
||||
lua_pushstring(L, "request_info");
|
||||
lua_newtable(L);
|
||||
reg_string(L, "request_method", conn->request_info.request_method);
|
||||
reg_string(L, "request_uri", conn->request_info.request_uri);
|
||||
@ -2534,6 +2532,15 @@ prepare_lua_request_info(struct mg_connection *conn, lua_State *L)
|
||||
reg_string(L, "finger", conn->request_info.client_cert->finger);
|
||||
lua_rawset(L, -3);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
static void
|
||||
prepare_lua_request_info(const struct mg_connection *conn, lua_State *L)
|
||||
{
|
||||
/* Export mg.request_info */
|
||||
lua_pushstring(L, "request_info");
|
||||
prepare_lua_request_info_inner(conn, L);
|
||||
|
||||
/* End of request_info */
|
||||
lua_rawset(L, -3);
|
||||
|
@ -2,18 +2,35 @@
|
||||
-- and "start" when the script is loading.
|
||||
shared.timer = 0
|
||||
|
||||
|
||||
function timer()
|
||||
-- Increment shared value.
|
||||
shared.timer = shared.timer + 1
|
||||
-- Return true to keep interval timer running or false to stop.
|
||||
-- Return true to keep interval timer running or false to stop.
|
||||
return true
|
||||
end
|
||||
|
||||
|
||||
function start()
|
||||
-- The "start" function is called when the server is ready.
|
||||
-- At this point, a timer can be created.
|
||||
-- At this point, a timer can be created.
|
||||
mg.set_interval(timer, 5)
|
||||
end
|
||||
|
||||
|
||||
function stop()
|
||||
-- The "stop" function is called when the server is stopping.
|
||||
end
|
||||
|
||||
function log(ri)
|
||||
-- The "log" function can be used to
|
||||
-- (a) filter messages and return boolean: true (log) or false (do not log)
|
||||
-- (b) format log message and return it as string (empty string will not log)
|
||||
-- (c) forward the log data to an external log
|
||||
|
||||
-- Example: Log only if the request was not made from localhost
|
||||
return (ri.remote_addr ~= "127.0.0.1");
|
||||
end
|
||||
|
||||
-- Return false to abort server startup.
|
||||
return true;
|
||||
|
Loading…
x
Reference in New Issue
Block a user