mod_log_forensic.c revision bf2c5978346829f62a64d3f2106a3cc08782b86c
/* 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
*
*
* 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:
*
* 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 "apr_strings.h"
#include "apr_atomic.h"
#include "http_protocol.h"
#include "test_char.h"
#include <unistd.h>
#endif
typedef struct fcfg {
const char *logname;
apr_file_t *fd;
} fcfg;
static apr_uint32_t next_id;
{
return cfg;
}
{
return cfg;
}
{
return 1;
return 0;
}
}
else {
APR_OS_DEFAULT, p)) != APR_SUCCESS) {
"could not open forensic log file %s.", fname);
return 0;
}
}
return 1;
}
server_rec *s)
{
for ( ; s ; s = s->next) {
if (!open_log(s, p)) {
return HTTP_INTERNAL_SERVER_ERROR;
}
}
return OK;
}
/* 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;
apr_pool_t *p;
} 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;
}
{
return 1;
}
{
/* note that we don't have to check h->pos here, coz its been done
for us by log_escape */
*h->pos++ = '|';
*h->pos++ = ':';
return 1;
}
static int log_before(request_rec *r)
{
const char *id;
hlog h;
apr_size_t n;
return DECLINED;
}
/* we make the assumption that we can't go through all the PIDs in
under 1 second */
}
h.p = r->pool;
h.count = 0;
*h.pos++ = '+';
*h.pos++ = '|';
*h.pos++ = '\n';
n = h.count-1;
return OK;
}
static int log_after(request_rec *r)
{
char *s;
apr_size_t l, n;
return DECLINED;
}
l = n = strlen(s);
return OK;
}
{
return NULL;
}
static const command_rec forensic_log_cmds[] =
{
"the filename of the forensic log"),
{ NULL }
};
static void register_hooks(apr_pool_t *p)
{
}
{
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 apr_table_t */
register_hooks /* register hooks */
};