⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 mod_log_forensic.c

📁 apache 安装教程 apache 安装教程
💻 C
字号:
/* Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements.  See the NOTICE file distributed with * this work for additional information regarding copyright ownership. * The ASF licenses this file to You under the Apache License, Version 2.0 * (the "License"); you may not use this file except in compliance with * the License.  You may obtain a copy of the License at * *     http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. *//* * See also support/check_forensic. * Relate the forensic log to the transfer log by including * %{forensic-id}n in the custom log format, for example: * CustomLog logs/custom "%h %l %u %t \"%r\" %>s %b %{forensic-id}n" * * Credit is due to Tina Bird <tbird@precision-guesswork.com>, whose * idea this module was. * *   Ben Laurie 29/12/2003 */#include "httpd.h"#include "http_config.h"#include "http_log.h"#include "multithread.h"#ifdef NETWARE#include "test_char.h"#else/* XXX This should be fixed in the INCLUDE path of the makefile   so that a specific location is not hard coded here. */#include "../../main/test_char.h"#endifmodule MODULE_VAR_EXPORT log_forensic_module;#ifdef WIN32static DWORD tls_index;BOOL WINAPI DllMain (HINSTANCE dllhandle, DWORD reason, LPVOID reserved){    switch (reason) {    case DLL_PROCESS_ATTACH:	tls_index = TlsAlloc();    case DLL_THREAD_ATTACH: /* intentional no break */	TlsSetValue(tls_index, 0);	break;    }    return TRUE;}const char * get_forensic_id(pool *p){    /* The 'error' default for Get undefined is 0 - a nice number     * for this purpose.  The cast might look evil, but the evil     * empire had switched this API out from underneath developers,     * and the DWORD flavor will truncate nicely for our purposes.     */    DWORD next_id = (DWORD)TlsGetValue(tls_index);    TlsSetValue(tls_index, (void*)(next_id + 1));    return ap_psprintf(p, "%x:%x:%lx:%x", GetCurrentProcessId(),                                           GetCurrentThreadId(),                                           time(NULL), next_id);}#else /* !WIN32 *//* Even when not MULTITHREAD, this will return a single structure, since * APACHE_TLS should be defined as empty on single-threaded platforms. */const char * get_forensic_id(pool *p){    static APACHE_TLS next_id = 0;    /* we make the assumption that we can't go through all the PIDs in       under 1 second */#ifdef MULTITHREAD    return ap_psprintf(p, "%x:%x:%lx:%x", getpid(), gettid(), time(NULL), next_id++);#else    return ap_psprintf(p, "%x:%lx:%x", getpid(), time(NULL), next_id++);#endif}#endif /* !WIN32 */typedef struct fcfg {    char *logname;    int fd;} fcfg;static void *make_forensic_log_scfg(pool *p, server_rec *s){    fcfg *cfg = ap_pcalloc(p, sizeof *cfg);    cfg->logname = NULL;    cfg->fd = -1;    return cfg;}static void *merge_forensic_log_scfg(pool *p, void *parent, void *new){    fcfg *cfg = ap_pcalloc(p, sizeof *cfg);    fcfg *pc = parent;    fcfg *nc = new;    cfg->logname = ap_pstrdup(p, nc->logname ? nc->logname : pc->logname);    cfg->fd = -1;    return cfg;}static void open_log(server_rec *s, pool *p){    fcfg *cfg = ap_get_module_config(s->module_config, &log_forensic_module);    if (!cfg->logname || cfg->fd >= 0)        return;    if (*cfg->logname == '|') {        piped_log *pl;        pl = ap_open_piped_log(p, cfg->logname+1);        if (pl == NULL) {            ap_log_error(APLOG_MARK, APLOG_ERR, s,                         "couldn't spawn forensic log pipe %s", cfg->logname);            exit(1);        }        cfg->fd = ap_piped_log_write_fd(pl);    }    else {        char *fname = ap_server_root_relative(p, cfg->logname);        if ((cfg->fd = ap_popenf_ex(p, fname, O_WRONLY | O_APPEND | O_CREAT,                                    0644, 1)) < 0) {            ap_log_error(APLOG_MARK, APLOG_ERR, s,                         "could not open forensic log file %s.", fname);            exit(1);        }    }}static void log_init(server_rec *s, pool *p){    for ( ; s ; s = s->next)        open_log(s, p);}  /* e is the first _invalid_ location in q   N.B. returns the terminating NUL. */static char *log_escape(char *q, const char *e, const char *p){    for ( ; *p ; ++p) {        ap_assert(q < e);        if (test_char_table[*(unsigned char *)p]&T_ESCAPE_FORENSIC) {            ap_assert(q+2 < e);            *q++ = '%';            sprintf(q, "%02x", *(unsigned char *)p);            q += 2;        }        else            *q++ = *p;    }    ap_assert(q < e);    *q = '\0';    return q;}typedef struct hlog {    char *log;    char *pos;    char *end;    pool *p;    int count;} hlog;static int count_string(const char *p){    int n;    for (n = 0 ; *p ; ++p, ++n)        if (test_char_table[*(unsigned char *)p]&T_ESCAPE_FORENSIC)            n += 2;    return n;}static int count_headers(void *h_, const char *key, const char *value){    hlog *h = h_;    h->count += count_string(key)+count_string(value)+2;    return 1;}static int log_headers(void *h_, const char *key, const char *value){    hlog *h = h_;    /* note that we don't have to check h->pos here, coz its been done       for us by log_escape */    *h->pos++ = '|';    h->pos = log_escape(h->pos, h->end, key);    *h->pos++ = ':';    h->pos = log_escape(h->pos, h->end, value);    return 1;}static int log_before(request_rec *r){    fcfg *cfg = ap_get_module_config(r->server->module_config,                                     &log_forensic_module);    const char *id;    hlog h;    if (cfg->fd < 0 || r->prev)        return DECLINED;    if (!(id = ap_table_get(r->subprocess_env, "UNIQUE_ID"))) {        id = get_forensic_id(r->pool);    }    h.p = r->pool;    h.count = 0;    ap_table_do(count_headers, &h, r->headers_in, NULL);    h.count += 1+strlen(id)+1+count_string(r->the_request)+1+1;    h.log = ap_palloc(r->pool, h.count);    h.pos = h.log;    h.end = h.log+h.count;    *h.pos++ = '+';    strcpy(h.pos, id);    h.pos += strlen(h.pos);    *h.pos++ = '|';    h.pos = log_escape(h.pos, h.end, r->the_request);    ap_table_do(log_headers, &h, r->headers_in, NULL);    ap_assert(h.pos < h.end);    *h.pos++ = '\n';    write(cfg->fd, h.log, h.count-1);    ap_table_setn(r->notes, "forensic-id", id);    return OK;}static int log_after(request_rec *r){    fcfg *cfg = ap_get_module_config(r->server->module_config,                                     &log_forensic_module);    const char *id;    char *s;    if(cfg->fd < 0)        return DECLINED;    id = ap_table_get(r->notes, "forensic-id");    if (!id)        return DECLINED;    s = ap_pstrcat(r->pool, "-", id, "\n", NULL);    write(cfg->fd, s, strlen(s));    return OK;}static const char *set_forensic_log(cmd_parms *cmd,  void *dummy,  char *fn){    fcfg *cfg = ap_get_module_config(cmd->server->module_config,                                     &log_forensic_module);    cfg->logname = fn;    return NULL;}static const command_rec forensic_log_cmds[] ={    { "ForensicLog",  set_forensic_log,  NULL,  RSRC_CONF,  TAKE1,      "the filename of the forensic log" },    { NULL }};module MODULE_VAR_EXPORT log_forensic_module ={    STANDARD_MODULE_STUFF,    log_init,                   /* initializer */    NULL,                       /* create per-dir config */    NULL,                       /* merge per-dir config */    make_forensic_log_scfg,     /* server config */    merge_forensic_log_scfg,    /* merge server config */    forensic_log_cmds,          /* command table */    NULL,                       /* handlers */    NULL,                       /* filename translation */    NULL,                       /* check_user_id */    NULL,                       /* check auth */    NULL,                       /* check access */    NULL,                       /* type_checker */    NULL,                       /* fixups */    log_after,                  /* logger */    NULL,                       /* header parser */    NULL,                       /* child_init */    NULL,                       /* child_exit */    log_before                  /* post read-request */};

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -