diff options
| author | 2002-04-21 23:01:53 +0000 | |
|---|---|---|
| committer | 2002-04-21 23:01:53 +0000 | |
| commit | b63c5d2438aabf0d7721c38387995cb4fb98345f (patch) | |
| tree | 73f99f046bb7daae711c5a7bdfe0f89677e22bda /mod_log_sql.c | |
| parent | 13e1ecc20c883f71b54bf1fe09488b028e51171e (diff) | |
Significant bugfixes and feature additions on the way to 1.16...
Diffstat (limited to 'mod_log_sql.c')
| -rw-r--r-- | mod_log_sql.c | 674 |
1 files changed, 383 insertions, 291 deletions
diff --git a/mod_log_sql.c b/mod_log_sql.c index 16deada..02b0dd7 100644 --- a/mod_log_sql.c +++ b/mod_log_sql.c | |||
| @@ -1,52 +1,72 @@ | |||
| 1 | /* $Id: mod_log_sql.c,v 1.8 2002/04/08 07:06:20 helios Exp $ */ | 1 | /* $Id: mod_log_sql.c,v 1.9 2002/04/21 23:01:53 helios Exp $ */ |
| 2 | 2 | ||
| 3 | /* --------* | ||
| 4 | * DEFINES * | ||
| 5 | * --------*/ | ||
| 3 | 6 | ||
| 4 | /* DEFINES */ | 7 | /* The enduser probably won't modify these */ |
| 5 | #define MYSQL_ERROR(mysql) ((mysql)?(mysql_error(mysql)):"MySQL server has gone away") | 8 | #define MYSQL_ERROR(mysql) ((mysql)?(mysql_error(mysql)):"MySQL server has gone away") |
| 6 | #define ERRLEVEL APLOG_ERR|APLOG_NOERRNO | 9 | #define ERRLEVEL APLOG_ERR|APLOG_NOERRNO |
| 7 | #define WARNINGLEVEL APLOG_WARNING|APLOG_NOERRNO | 10 | #define WARNINGLEVEL APLOG_WARNING|APLOG_NOERRNO |
| 8 | #define NOTICELEVEL APLOG_NOTICE|APLOG_NOERRNO | 11 | #define NOTICELEVEL APLOG_NOTICE|APLOG_NOERRNO |
| 9 | #define DEBUGLEVEL APLOG_INFO|APLOG_NOERRNO | 12 | #define DEBUGLEVEL APLOG_DEBUG|APLOG_NOERRNO |
| 10 | #define PRESERVEFILE "/tmp/mysql-preserve" | ||
| 11 | /* (MYSQLSOCKET, DEBUG and WANT_SSL_LOGGING are defined in the Makefile DEFS line.) */ | ||
| 12 | 13 | ||
| 14 | /* The enduser may wish to modify these */ | ||
| 15 | #define WANT_SSL_LOGGING | ||
| 16 | #undef DEBUG | ||
| 13 | 17 | ||
| 14 | 18 | ||
| 15 | /* INCLUDES */ | 19 | /* ---------* |
| 20 | * INCLUDES * | ||
| 21 | * ---------*/ | ||
| 16 | #include <time.h> | 22 | #include <time.h> |
| 17 | #include <mysql/mysql.h> | 23 | #include <mysql/mysql.h> |
| 18 | #include <stdio.h> | 24 | #include <stdio.h> |
| 19 | |||
| 20 | #include "httpd.h" | 25 | #include "httpd.h" |
| 21 | #include "http_config.h" | 26 | #include "http_config.h" |
| 22 | #include "http_log.h" | 27 | #include "http_log.h" |
| 23 | #include "http_core.h" | 28 | #include "http_core.h" |
| 24 | 29 | ||
| 25 | /* M_M_N is defined in /usr/local/Apache/include/ap_mmn.h, 19990320 as of this writing. */ | 30 | #if MODULE_MAGIC_NUMBER >= 19980324 /* M_M_N is defined in /usr/local/Apache/include/ap_mmn.h, 19990320 as of this writing. */ |
| 26 | #if MODULE_MAGIC_NUMBER >= 19980324 /* 1.3b6 or later */ | ||
| 27 | #include "ap_compat.h" | 31 | #include "ap_compat.h" |
| 28 | #endif | 32 | #endif |
| 29 | 33 | ||
| 30 | #ifdef WANT_SSL_LOGGING /* Defined in Makefile */ | 34 | #ifdef WANT_SSL_LOGGING |
| 31 | #include "mod_ssl.h" | 35 | #include "mod_ssl.h" |
| 32 | #endif | 36 | #endif |
| 33 | 37 | ||
| 34 | 38 | ||
| 39 | /* -------------* | ||
| 40 | * DECLARATIONS * | ||
| 41 | * -------------*/ | ||
| 35 | 42 | ||
| 36 | /* DECLARATIONS */ | 43 | /* Declare ourselves so the configuration routines can find and know us. */ |
| 37 | module mysql_log_module; | 44 | module mysql_log_module; |
| 38 | 45 | ||
| 46 | /* The contents of these are known 'Apache wide' and are not variable | ||
| 47 | * on a per-virtual-server basis. Every virtual server 'knows' the | ||
| 48 | * same versions of these variables. | ||
| 49 | */ | ||
| 39 | MYSQL sql_server, *mysql_log = NULL; | 50 | MYSQL sql_server, *mysql_log = NULL; |
| 40 | 51 | ||
| 52 | int massvirtual = 0; | ||
| 41 | char *db_name = NULL; | 53 | char *db_name = NULL; |
| 42 | char *db_host = NULL; | 54 | char *db_host = NULL; |
| 43 | char *db_user = NULL; | 55 | char *db_user = NULL; |
| 44 | char *db_pwd = NULL; | 56 | char *db_pwd = NULL; |
| 45 | char *cookie_name = NULL; | 57 | char *socket_file = "/var/lib/mysql/mysql.sock"; |
| 46 | 58 | ||
| 47 | typedef const char *(*item_key_func) (request_rec *, char *); | 59 | typedef const char *(*item_key_func) (request_rec *, char *); |
| 48 | 60 | ||
| 61 | /* But the contents of this structure will vary by virtual server. | ||
| 62 | * This permits each virtual server to vary its configuration slightly | ||
| 63 | * for per-server customization. | ||
| 64 | * | ||
| 65 | * Each child process has its own segregated copy of this structure. | ||
| 66 | */ | ||
| 49 | typedef struct { | 67 | typedef struct { |
| 68 | int create_tables; | ||
| 69 | int table_made; | ||
| 50 | char *referer_table_name; | 70 | char *referer_table_name; |
| 51 | char *agent_table_name; | 71 | char *agent_table_name; |
| 52 | char *transfer_table_name; | 72 | char *transfer_table_name; |
| @@ -54,11 +74,14 @@ typedef struct { | |||
| 54 | array_header *transfer_ignore_list; | 74 | array_header *transfer_ignore_list; |
| 55 | array_header *remhost_ignore_list; | 75 | array_header *remhost_ignore_list; |
| 56 | char *transfer_log_format; | 76 | char *transfer_log_format; |
| 77 | char *preserve_file; | ||
| 78 | char *cookie_name; | ||
| 57 | } log_mysql_state; | 79 | } log_mysql_state; |
| 58 | 80 | ||
| 59 | 81 | ||
| 60 | 82 | /* -----------------* | |
| 61 | /* FUNCTIONS */ | 83 | * HELPER FUNCTIONS * |
| 84 | * -----------------*/ | ||
| 62 | static char *format_integer(pool *p, int i) | 85 | static char *format_integer(pool *p, int i) |
| 63 | { | 86 | { |
| 64 | char dummy[40]; | 87 | char dummy[40]; |
| @@ -110,7 +133,7 @@ static const char *extract_ssl_keysize(request_rec *r, char *a) | |||
| 110 | if (ap_ctx_get(r->connection->client->ctx, "ssl") != NULL) { | 133 | if (ap_ctx_get(r->connection->client->ctx, "ssl") != NULL) { |
| 111 | result = ssl_var_lookup(r->pool, r->server, r->connection, r, "SSL_CIPHER_USEKEYSIZE"); | 134 | result = ssl_var_lookup(r->pool, r->server, r->connection, r, "SSL_CIPHER_USEKEYSIZE"); |
| 112 | #ifdef DEBUG | 135 | #ifdef DEBUG |
| 113 | ap_log_error(APLOG_MARK,DEBUGLEVEL,r->server,"mod_log_mysql: SSL_KEYSIZE: %s", result); | 136 | ap_log_error(APLOG_MARK,DEBUGLEVEL,r->server,"SSL_KEYSIZE: %s", result); |
| 114 | #endif | 137 | #endif |
| 115 | if (result != NULL && result[0] == '\0') | 138 | if (result != NULL && result[0] == '\0') |
| 116 | result = NULL; | 139 | result = NULL; |
| @@ -127,7 +150,7 @@ static const char *extract_ssl_maxkeysize(request_rec *r, char *a) | |||
| 127 | if (ap_ctx_get(r->connection->client->ctx, "ssl") != NULL) { | 150 | if (ap_ctx_get(r->connection->client->ctx, "ssl") != NULL) { |
| 128 | result = ssl_var_lookup(r->pool, r->server, r->connection, r, "SSL_CIPHER_ALGKEYSIZE"); | 151 | result = ssl_var_lookup(r->pool, r->server, r->connection, r, "SSL_CIPHER_ALGKEYSIZE"); |
| 129 | #ifdef DEBUG | 152 | #ifdef DEBUG |
| 130 | ap_log_error(APLOG_MARK,DEBUGLEVEL,r->server,"mod_log_mysql: SSL_ALGKEYSIZE: %s", result); | 153 | ap_log_error(APLOG_MARK,DEBUGLEVEL,r->server,"SSL_ALGKEYSIZE: %s", result); |
| 131 | #endif | 154 | #endif |
| 132 | if (result != NULL && result[0] == '\0') | 155 | if (result != NULL && result[0] == '\0') |
| 133 | result = NULL; | 156 | result = NULL; |
| @@ -144,7 +167,7 @@ static const char *extract_ssl_cipher(request_rec *r, char *a) | |||
| 144 | if (ap_ctx_get(r->connection->client->ctx, "ssl") != NULL) { | 167 | if (ap_ctx_get(r->connection->client->ctx, "ssl") != NULL) { |
| 145 | result = ssl_var_lookup(r->pool, r->server, r->connection, r, "SSL_CIPHER"); | 168 | result = ssl_var_lookup(r->pool, r->server, r->connection, r, "SSL_CIPHER"); |
| 146 | #ifdef DEBUG | 169 | #ifdef DEBUG |
| 147 | ap_log_error(APLOG_MARK,DEBUGLEVEL,r->server,"mod_log_mysql: SSL_CIPHER: %s", result); | 170 | ap_log_error(APLOG_MARK,DEBUGLEVEL,r->server,"SSL_CIPHER: %s", result); |
| 148 | #endif | 171 | #endif |
| 149 | if (result != NULL && result[0] == '\0') | 172 | if (result != NULL && result[0] == '\0') |
| 150 | result = NULL; | 173 | result = NULL; |
| @@ -299,15 +322,27 @@ static const char *extract_cookie(request_rec *r, char *a) | |||
| 299 | char *isvalid; | 322 | char *isvalid; |
| 300 | char *cookiebuf; | 323 | char *cookiebuf; |
| 301 | 324 | ||
| 325 | log_mysql_state *cls = get_module_config(r->server->module_config, &mysql_log_module); | ||
| 326 | |||
| 327 | #ifdef DEBUG | ||
| 328 | ap_log_error(APLOG_MARK,DEBUGLEVEL,r->server,"watching for cookie '%s'", cls->cookie_name); | ||
| 329 | #endif | ||
| 330 | |||
| 331 | /* Fetch out the cookie header */ | ||
| 302 | cookiestr = (char *)table_get(r->headers_in, "cookie2"); | 332 | cookiestr = (char *)table_get(r->headers_in, "cookie2"); |
| 303 | if (cookiestr != NULL) { | 333 | if (cookiestr != NULL) { |
| 304 | #ifdef DEBUG | 334 | #ifdef DEBUG |
| 305 | ap_log_error(APLOG_MARK,DEBUGLEVEL,r->server,"mod_log_mysql: Cookie2: [%s]", cookiestr); | 335 | ap_log_error(APLOG_MARK,DEBUGLEVEL,r->server,"Cookie2: [%s]", cookiestr); |
| 306 | #endif | 336 | #endif |
| 307 | isvalid = strstr(cookiestr, cookie_name); | 337 | /* Does the cookie string contain one with our name? */ |
| 338 | isvalid = strstr(cookiestr, cls->cookie_name); | ||
| 308 | if (isvalid != NULL) { | 339 | if (isvalid != NULL) { |
| 309 | isvalid += strlen(cookie_name) + 1; | 340 | /* Move past the cookie name and equal sign */ |
| 341 | isvalid += strlen(cls->cookie_name) + 1; | ||
| 342 | /* Duplicate it into the pool */ | ||
| 310 | cookiebuf = ap_pstrdup(r->pool, isvalid); | 343 | cookiebuf = ap_pstrdup(r->pool, isvalid); |
| 344 | /* Segregate just this cookie out of the string | ||
| 345 | * with a terminating nul at the first semicolon */ | ||
| 311 | cookieend = strchr(cookiebuf, ';'); | 346 | cookieend = strchr(cookiebuf, ';'); |
| 312 | if (cookieend != NULL) | 347 | if (cookieend != NULL) |
| 313 | *cookieend = '\0'; | 348 | *cookieend = '\0'; |
| @@ -318,11 +353,11 @@ static const char *extract_cookie(request_rec *r, char *a) | |||
| 318 | cookiestr = (char *)table_get(r->headers_in, "cookie"); | 353 | cookiestr = (char *)table_get(r->headers_in, "cookie"); |
| 319 | if (cookiestr != NULL) { | 354 | if (cookiestr != NULL) { |
| 320 | #ifdef DEBUG | 355 | #ifdef DEBUG |
| 321 | ap_log_error(APLOG_MARK,DEBUGLEVEL,r->server,"mod_log_mysql: Cookie: [%s]", cookiestr); | 356 | ap_log_error(APLOG_MARK,DEBUGLEVEL,r->server,"Cookie: [%s]", cookiestr); |
| 322 | #endif | 357 | #endif |
| 323 | isvalid = strstr(cookiestr, cookie_name); | 358 | isvalid = strstr(cookiestr, cls->cookie_name); |
| 324 | if (isvalid != NULL) { | 359 | if (isvalid != NULL) { |
| 325 | isvalid += strlen(cookie_name) + 1; | 360 | isvalid += strlen(cls->cookie_name) + 1; |
| 326 | cookiebuf = ap_pstrdup(r->pool, isvalid); | 361 | cookiebuf = ap_pstrdup(r->pool, isvalid); |
| 327 | cookieend = strchr(cookiebuf, ';'); | 362 | cookieend = strchr(cookiebuf, ';'); |
| 328 | if (cookieend != NULL) | 363 | if (cookieend != NULL) |
| @@ -334,11 +369,11 @@ static const char *extract_cookie(request_rec *r, char *a) | |||
| 334 | cookiestr = table_get(r->headers_out, "set-cookie"); | 369 | cookiestr = table_get(r->headers_out, "set-cookie"); |
| 335 | if (cookiestr != NULL) { | 370 | if (cookiestr != NULL) { |
| 336 | #ifdef DEBUG | 371 | #ifdef DEBUG |
| 337 | ap_log_error(APLOG_MARK,DEBUGLEVEL,r->server,"mod_log_mysql: Set-Cookie: [%s]", cookiestr); | 372 | ap_log_error(APLOG_MARK,DEBUGLEVEL,r->server,"Set-Cookie: [%s]", cookiestr); |
| 338 | #endif | 373 | #endif |
| 339 | isvalid = strstr(cookiestr, cookie_name); | 374 | isvalid = strstr(cookiestr, cls->cookie_name); |
| 340 | if (isvalid != NULL) { | 375 | if (isvalid != NULL) { |
| 341 | isvalid += strlen(cookie_name) + 1; | 376 | isvalid += strlen(cls->cookie_name) + 1; |
| 342 | cookiebuf = ap_pstrdup(r->pool, isvalid); | 377 | cookiebuf = ap_pstrdup(r->pool, isvalid); |
| 343 | cookieend = strchr(cookiebuf, ';'); | 378 | cookieend = strchr(cookiebuf, ';'); |
| 344 | if (cookieend != NULL) | 379 | if (cookieend != NULL) |
| @@ -350,23 +385,6 @@ static const char *extract_cookie(request_rec *r, char *a) | |||
| 350 | return "-"; | 385 | return "-"; |
| 351 | } | 386 | } |
| 352 | 387 | ||
| 353 | /* | ||
| 354 | static const char *extract_forwarded(request_rec *r, char *a) | ||
| 355 | { | ||
| 356 | return table_get(r->subprocess_env, "HTTP_FORWARDED"); | ||
| 357 | } | ||
| 358 | |||
| 359 | static const char *extract_via(request_rec *r, char *a) | ||
| 360 | { | ||
| 361 | return table_get(r->subprocess_env, "HTTP_VIA"); | ||
| 362 | } | ||
| 363 | |||
| 364 | static const char *extract_forwarded_for(request_rec *r, char *a) | ||
| 365 | { | ||
| 366 | return table_get(r->subprocess_env, "HTTP_X_FORWARDED_FOR"); | ||
| 367 | } | ||
| 368 | */ | ||
| 369 | |||
| 370 | static const char *extract_request_timestamp(request_rec *r, char *a) | 388 | static const char *extract_request_timestamp(request_rec *r, char *a) |
| 371 | { | 389 | { |
| 372 | char tstr[32]; | 390 | char tstr[32]; |
| @@ -390,11 +408,11 @@ static const char *extract_env_var(request_rec *r, char *a) | |||
| 390 | 408 | ||
| 391 | 409 | ||
| 392 | struct log_mysql_item_list { | 410 | struct log_mysql_item_list { |
| 393 | char ch; | 411 | char ch; /* its letter code */ |
| 394 | item_key_func func; | 412 | item_key_func func; /* its extraction function */ |
| 395 | const char *sql_field_name; | 413 | const char *sql_field_name; /* its column in SQL */ |
| 396 | int want_orig_default; | 414 | int want_orig_default; /* if it requires the original request prior to internal redirection */ |
| 397 | int string_contents; | 415 | int string_contents; /* if it returns a string */ |
| 398 | } log_mysql_item_keys[] = { | 416 | } log_mysql_item_keys[] = { |
| 399 | 417 | ||
| 400 | { 'A', extract_agent, "agent", 1, 1 }, | 418 | { 'A', extract_agent, "agent", 1, 1 }, |
| @@ -420,10 +438,6 @@ struct log_mysql_item_list { | |||
| 420 | { 'u', extract_remote_user, "remote_user", 0, 1 }, | 438 | { 'u', extract_remote_user, "remote_user", 0, 1 }, |
| 421 | { 'U', extract_request_uri, "request_uri", 1, 1 }, | 439 | { 'U', extract_request_uri, "request_uri", 1, 1 }, |
| 422 | { 'v', extract_virtual_host, "virtual_host", 0, 1 }, | 440 | { 'v', extract_virtual_host, "virtual_host", 0, 1 }, |
| 423 | /* { 'V', extract_via, "via", 0, 1 }, | ||
| 424 | { 'w', extract_forwarded, "forwarded", 0, 1 }, | ||
| 425 | { 'W', extract_forwarded_for, "forwarded_for", 0, 1 }, | ||
| 426 | */ | ||
| 427 | #ifdef WANT_SSL_LOGGING | 441 | #ifdef WANT_SSL_LOGGING |
| 428 | { 'q', extract_ssl_keysize, "ssl_keysize", 0, 1 }, | 442 | { 'q', extract_ssl_keysize, "ssl_keysize", 0, 1 }, |
| 429 | { 'Q', extract_ssl_maxkeysize, "ssl_maxkeysize", 0, 1 }, | 443 | { 'Q', extract_ssl_maxkeysize, "ssl_maxkeysize", 0, 1 }, |
| @@ -436,66 +450,54 @@ struct log_mysql_item_list { | |||
| 436 | /* Routine to escape the 'dangerous' characters that would otherwise | 450 | /* Routine to escape the 'dangerous' characters that would otherwise |
| 437 | * corrupt the INSERT string: ', \, and " | 451 | * corrupt the INSERT string: ', \, and " |
| 438 | */ | 452 | */ |
| 439 | const char *mysql_escape_log(const char *str, pool *p) | 453 | const char *escape_query(const char *from_str, pool *p) |
| 440 | { | 454 | { |
| 441 | register int i = 0, j = 0; | 455 | if (!from_str) |
| 442 | int need_to_escape = 0; | ||
| 443 | |||
| 444 | if (!str) { | ||
| 445 | return NULL; | 456 | return NULL; |
| 446 | } | 457 | else { |
| 447 | 458 | char *to_str; | |
| 448 | /* First find out if we need to escape. */ | 459 | unsigned long length = strlen(from_str); |
| 449 | i = 0; | 460 | unsigned long retval; |
| 450 | while (str[i]) { | 461 | |
| 451 | /* WAS THIS WRONG in 1.05?!? if (str[i] != '\'' || str[i] != '\\' || str[i] != '\"') { */ | ||
| 452 | if (str[i] == '\'' || str[i] == '\\' || str[i] == '\"') { | ||
| 453 | need_to_escape = 1; | ||
| 454 | break; | ||
| 455 | } | ||
| 456 | i++; | ||
| 457 | } | ||
| 458 | |||
| 459 | if (need_to_escape) { | ||
| 460 | char *tmp_str; | ||
| 461 | int length = strlen(str); | ||
| 462 | |||
| 463 | /* Pre-allocate a new string that could hold twice the original, which would only | 462 | /* Pre-allocate a new string that could hold twice the original, which would only |
| 464 | * happen if the whole original string was 'dangerous' characters. | 463 | * happen if the whole original string was 'dangerous' characters. |
| 465 | */ | 464 | */ |
| 466 | tmp_str = (char *) palloc(p, length * 2 + 1); | 465 | to_str = (char *) ap_palloc(p, length * 2 + 1); |
| 467 | if (!tmp_str) { | 466 | if (!to_str) { |
| 468 | return str; | 467 | return from_str; |
| 469 | } | 468 | } |
| 470 | 469 | ||
| 471 | /* Walk through character-by-character, escaping any dangerous characters found. */ | 470 | if (!mysql_log) { |
| 472 | for (i = 0, j = 0; i < length; i++, j++) { | 471 | /* Well, I would have liked to use the current database charset. mysql is |
| 473 | switch (str[i]) { | 472 | * unavailable, however, so I fall back to the slightly less respectful |
| 474 | case '\'': | 473 | * mysql_escape_string() function that uses the default charset. |
| 475 | case '\"': | 474 | */ |
| 476 | case '\\': | 475 | retval = mysql_escape_string(to_str, from_str, length); |
| 477 | tmp_str[j] = '\\'; | 476 | } else { |
| 478 | j++; | 477 | /* MySQL is available, so I'll go ahead and respect the current charset when |
| 479 | default: | 478 | * I perform the escape. |
| 480 | tmp_str[j] = str[i]; | 479 | */ |
| 481 | } | 480 | retval = mysql_real_escape_string(mysql_log, to_str, from_str, length); |
| 482 | } | 481 | } |
| 483 | tmp_str[j] = '\0'; | 482 | |
| 484 | return tmp_str; | 483 | if (retval) |
| 485 | } else { | 484 | return to_str; |
| 486 | return str; | 485 | else |
| 486 | return from_str; | ||
| 487 | } | 487 | } |
| 488 | } | 488 | } |
| 489 | 489 | ||
| 490 | int open_logdb_link() | 490 | int open_logdb_link() |
| 491 | { | 491 | { |
| 492 | /* Returns 2 if already connected, 1 if successful, 0 if unsuccessful */ | ||
| 493 | |||
| 492 | if (mysql_log != NULL) { | 494 | if (mysql_log != NULL) { |
| 493 | return 2; | 495 | return 2; |
| 494 | } | 496 | } |
| 495 | 497 | ||
| 496 | if (db_name) { | 498 | if (db_name) { |
| 497 | mysql_init(&sql_server); | 499 | mysql_init(&sql_server); |
| 498 | mysql_log = mysql_real_connect(&sql_server, db_host, db_user, db_pwd, db_name, 0, MYSQLSOCKET, 0); | 500 | mysql_log = mysql_real_connect(&sql_server, db_host, db_user, db_pwd, db_name, 0, socket_file, 0); |
| 499 | 501 | ||
| 500 | if (mysql_log != NULL) { | 502 | if (mysql_log != NULL) { |
| 501 | return 1; | 503 | return 1; |
| @@ -509,18 +511,27 @@ int open_logdb_link() | |||
| 509 | void preserve_entry(request_rec *r, const char *query) | 511 | void preserve_entry(request_rec *r, const char *query) |
| 510 | { | 512 | { |
| 511 | FILE *fp; | 513 | FILE *fp; |
| 514 | log_mysql_state *cls = get_module_config(r->server->module_config, &mysql_log_module); | ||
| 512 | 515 | ||
| 513 | fp = fopen(PRESERVEFILE, "a"); | 516 | fp = pfopen(r->pool, cls->preserve_file, "a"); |
| 514 | if (fp == NULL) | 517 | if (fp == NULL) |
| 515 | ap_log_error(APLOG_MARK,ERRLEVEL,r->server,"MySQL: attempted append of local offline file but failed."); | 518 | ap_log_error(APLOG_MARK,ERRLEVEL,r->server,"attempted append of local offline file but failed."); |
| 516 | else | 519 | else |
| 517 | fprintf(fp,"%s;\n", query); | 520 | fprintf(fp,"%s;\n", query); |
| 518 | fclose(fp); | 521 | pfclose(r->pool, fp); |
| 519 | } | 522 | } |
| 520 | 523 | ||
| 524 | /*-----------------------------------------------------*/ | ||
| 525 | /* safe_mysql_query: perform a database insert with */ | ||
| 526 | /* a degree of safety and error checking. */ | ||
| 527 | /* */ | ||
| 528 | /* Parms: request record, SQL insert statement */ | ||
| 529 | /* Returns: 0 (OK) on success */ | ||
| 530 | /* mysql return code on error */ | ||
| 531 | /*-----------------------------------------------------*/ | ||
| 521 | int safe_mysql_query(request_rec *r, const char *query) | 532 | int safe_mysql_query(request_rec *r, const char *query) |
| 522 | { | 533 | { |
| 523 | int retval = 1; | 534 | int retval; |
| 524 | struct timespec delay, remainder; | 535 | struct timespec delay, remainder; |
| 525 | int ret; | 536 | int ret; |
| 526 | char *str; | 537 | char *str; |
| @@ -539,19 +550,22 @@ int safe_mysql_query(request_rec *r, const char *query) | |||
| 539 | * at any time, hence the check. */ | 550 | * at any time, hence the check. */ |
| 540 | if ( retval != 0 ) | 551 | if ( retval != 0 ) |
| 541 | { | 552 | { |
| 553 | log_mysql_state *cls = get_module_config(r->server->module_config, &mysql_log_module); | ||
| 554 | |||
| 542 | /* Something went wrong, so start by trying to restart the db link. */ | 555 | /* Something went wrong, so start by trying to restart the db link. */ |
| 543 | ap_log_error(APLOG_MARK,ERRLEVEL,r->server,"MySQL: attempting reconnect because API said: %s", MYSQL_ERROR(mysql_log)); | 556 | ap_log_error(APLOG_MARK,ERRLEVEL,r->server,"attempting reconnect because API said: %s", mysql_error(mysql_log)); |
| 544 | 557 | ||
| 545 | mysql_log = NULL; | 558 | mysql_log = NULL; |
| 546 | open_logdb_link(); | 559 | open_logdb_link(); |
| 547 | 560 | ||
| 548 | if (mysql_log == NULL) { /* still unable to link */ | 561 | if (mysql_log == NULL) { /* still unable to link */ |
| 549 | signal(SIGPIPE, handler); | 562 | signal(SIGPIPE, handler); |
| 550 | ap_log_error(APLOG_MARK,ERRLEVEL,r->server,"MySQL: httpd child reconnect failed, unable to reach database. SQL logging stopped until an httpd child regains a db connection."); | 563 | ap_log_error(APLOG_MARK,ERRLEVEL,r->server,"httpd child reconnect failed, unable to reach database. SQL logging stopped until an httpd child regains a db connection."); |
| 551 | ap_log_error(APLOG_MARK,ERRLEVEL,r->server,"MySQL: log entries are being preserved in %s",PRESERVEFILE); | 564 | ap_log_error(APLOG_MARK,ERRLEVEL,r->server,"log entries are being preserved in %s", cls->preserve_file); |
| 565 | preserve_entry(r, query); | ||
| 552 | return retval; | 566 | return retval; |
| 553 | } else { | 567 | } else { |
| 554 | ap_log_error(APLOG_MARK,ERRLEVEL,r->server,"MySQL: reconnect successful."); | 568 | ap_log_error(APLOG_MARK,ERRLEVEL,r->server,"reconnect successful."); |
| 555 | } | 569 | } |
| 556 | 570 | ||
| 557 | /* Attempt a single re-try... First sleep for a tiny amount of time. */ | 571 | /* Attempt a single re-try... First sleep for a tiny amount of time. */ |
| @@ -559,7 +573,7 @@ int safe_mysql_query(request_rec *r, const char *query) | |||
| 559 | delay.tv_nsec = 500000000; /* max is 999999999 (nine nines) */ | 573 | delay.tv_nsec = 500000000; /* max is 999999999 (nine nines) */ |
| 560 | ret = nanosleep(&delay, &remainder); | 574 | ret = nanosleep(&delay, &remainder); |
| 561 | if (ret && errno != EINTR) | 575 | if (ret && errno != EINTR) |
| 562 | perror("nanosleep"); | 576 | ap_log_error(APLOG_MARK,ERRLEVEL,r->server,"nanosleep unsuccessful."); |
| 563 | 577 | ||
| 564 | /* Now make our second attempt */ | 578 | /* Now make our second attempt */ |
| 565 | retval = mysql_query(mysql_log,query); | 579 | retval = mysql_query(mysql_log,query); |
| @@ -567,13 +581,13 @@ int safe_mysql_query(request_rec *r, const char *query) | |||
| 567 | /* If this one also failed, log that and append to our local offline file */ | 581 | /* If this one also failed, log that and append to our local offline file */ |
| 568 | if ( retval != 0 ) | 582 | if ( retval != 0 ) |
| 569 | { | 583 | { |
| 570 | str = pstrcat(r->pool, "MySQL delayed insert attempt failed, API said: ", MYSQL_ERROR(mysql_log), NULL); | 584 | str = ap_pstrcat(r->pool, "delayed insert attempt failed, API said: ", MYSQL_ERROR(mysql_log), NULL); |
| 571 | ap_log_error(APLOG_MARK,ERRLEVEL,r->server,str); | 585 | ap_log_error(APLOG_MARK,ERRLEVEL,r->server,str); |
| 572 | 586 | ||
| 573 | preserve_entry(r, query); | 587 | preserve_entry(r, query); |
| 574 | ap_log_error(APLOG_MARK,ERRLEVEL,r->server,"MySQL: entry preserved in %s",PRESERVEFILE); | 588 | ap_log_error(APLOG_MARK,ERRLEVEL,r->server,"entry preserved in %s", cls->preserve_file); |
| 575 | } else { | 589 | } else { |
| 576 | ap_log_error(APLOG_MARK,ERRLEVEL,r->server,"MySQL: insert successful after a delayed retry."); | 590 | ap_log_error(APLOG_MARK,ERRLEVEL,r->server,"insert successful after a delayed retry."); |
| 577 | } | 591 | } |
| 578 | } | 592 | } |
| 579 | 593 | ||
| @@ -584,52 +598,44 @@ int safe_mysql_query(request_rec *r, const char *query) | |||
| 584 | } | 598 | } |
| 585 | 599 | ||
| 586 | 600 | ||
| 601 | /* ------------------------------------------------* | ||
| 602 | * Command handlers that are called according * | ||
| 603 | * to the directives found at Apache runtime. * | ||
| 604 | * ------------------------------------------------*/ | ||
| 587 | 605 | ||
| 588 | const char *set_referer_log_mysql_table(cmd_parms *parms, void *dummy, char *arg) | 606 | const char *set_massvirtual(cmd_parms *parms, void *dummy, int flag) |
| 589 | { | 607 | { |
| 590 | log_mysql_state *cls = get_module_config(parms->server->module_config, &mysql_log_module); | 608 | massvirtual = ( flag ? 1 : 0); |
| 591 | |||
| 592 | cls->referer_table_name = arg; | ||
| 593 | return NULL; | 609 | return NULL; |
| 594 | } | 610 | } |
| 595 | 611 | ||
| 596 | 612 | const char *set_log_mysql_create(cmd_parms *parms, void *dummy, int flag) | |
| 597 | const char *set_agent_log_mysql_table(cmd_parms *parms, void *dummy, char *arg) | ||
| 598 | { | 613 | { |
| 599 | log_mysql_state *cls = get_module_config(parms->server->module_config, &mysql_log_module); | 614 | log_mysql_state *cls = get_module_config(parms->server->module_config, &mysql_log_module); |
| 600 | 615 | ||
| 601 | cls->agent_table_name = arg; | 616 | cls->create_tables = ( flag ? 1 : 0); |
| 602 | return NULL; | 617 | return NULL; |
| 603 | } | 618 | } |
| 604 | 619 | ||
| 605 | 620 | const char *set_log_mysql_db(cmd_parms *parms, void *dummy, char *arg) | |
| 606 | const char *set_transfer_log_mysql_table(cmd_parms *parms, void *dummy, char *arg) | ||
| 607 | { | 621 | { |
| 608 | log_mysql_state *cls = get_module_config(parms->server->module_config, &mysql_log_module); | 622 | db_name = arg; |
| 609 | |||
| 610 | cls->transfer_table_name = arg; | ||
| 611 | return NULL; | 623 | return NULL; |
| 612 | } | 624 | } |
| 613 | 625 | ||
| 614 | 626 | const char *set_log_mysql_cookie(cmd_parms *parms, void *dummy, char *arg) | |
| 615 | const char *set_transfer_log_format(cmd_parms *parms, void *dummy, char *arg) | ||
| 616 | { | 627 | { |
| 617 | log_mysql_state *cls = get_module_config(parms->server->module_config, &mysql_log_module); | 628 | log_mysql_state *cls = get_module_config(parms->server->module_config, &mysql_log_module); |
| 618 | 629 | ||
| 619 | cls->transfer_log_format = arg; | 630 | cls->cookie_name = arg; |
| 620 | return NULL; | 631 | return NULL; |
| 621 | } | 632 | } |
| 622 | 633 | ||
| 623 | 634 | const char *set_log_mysql_preserve_file(cmd_parms *parms, void *dummy, char *arg) | |
| 624 | const char *set_log_mysql_db(cmd_parms *parms, void *dummy, char *arg) | ||
| 625 | { | 635 | { |
| 626 | db_name = arg; | 636 | log_mysql_state *cls = get_module_config(parms->server->module_config, &mysql_log_module); |
| 627 | return NULL; | ||
| 628 | } | ||
| 629 | 637 | ||
| 630 | const char *set_log_mysql_cookie(cmd_parms *parms, void *dummy, char *arg) | 638 | cls->preserve_file = arg; |
| 631 | { | ||
| 632 | cookie_name = arg; | ||
| 633 | return NULL; | 639 | return NULL; |
| 634 | } | 640 | } |
| 635 | 641 | ||
| @@ -647,12 +653,59 @@ const char *set_log_mysql_info(cmd_parms *parms, void *dummy, char *host, char * | |||
| 647 | return NULL; | 653 | return NULL; |
| 648 | } | 654 | } |
| 649 | 655 | ||
| 656 | const char *set_transfer_log_mysql_table(cmd_parms *parms, void *dummy, char *arg) | ||
| 657 | { | ||
| 658 | log_mysql_state *cls = get_module_config(parms->server->module_config, &mysql_log_module); | ||
| 659 | |||
| 660 | if (massvirtual == 1) { | ||
| 661 | char *base = "access_"; | ||
| 662 | char *tablename; | ||
| 663 | int i; | ||
| 664 | |||
| 665 | /* Find memory long enough to hold the table name + \0. */ | ||
| 666 | /* old way: */ | ||
| 667 | /* tablename = (char*)ap_palloc(parms->pool, (strlen(base) + strlen(parms->server->server_hostname) + 1) * sizeof(char));*/ | ||
| 668 | /* strcpy(tablename, base);*/ | ||
| 669 | /* strcat(tablename, parms->server->server_hostname);*/ | ||
| 670 | |||
| 671 | tablename = ap_pstrcat(parms->pool, base, parms->server->server_hostname, NULL); | ||
| 672 | |||
| 673 | /* Transform any dots to underscores */ | ||
| 674 | for (i = 0; i < strlen(tablename); i++) { | ||
| 675 | if (tablename[i] == '.') | ||
| 676 | tablename[i] = '_'; | ||
| 677 | } | ||
| 678 | |||
| 679 | /* Tell this virtual server its transfer table name, and | ||
| 680 | * turn on create_tables, which is implied by massvirtual. | ||
| 681 | */ | ||
| 682 | cls->transfer_table_name = tablename; | ||
| 683 | cls->create_tables = 1; | ||
| 684 | } else { | ||
| 685 | cls->transfer_table_name = arg; | ||
| 686 | } | ||
| 687 | return NULL; | ||
| 688 | } | ||
| 689 | |||
| 690 | const char *set_transfer_log_format(cmd_parms *parms, void *dummy, char *arg) | ||
| 691 | { | ||
| 692 | log_mysql_state *cls = get_module_config(parms->server->module_config, &mysql_log_module); | ||
| 693 | |||
| 694 | cls->transfer_log_format = arg; | ||
| 695 | return NULL; | ||
| 696 | } | ||
| 697 | |||
| 698 | const char *set_mysql_socket_file(cmd_parms *parms, void *dummy, char *arg) | ||
| 699 | { | ||
| 700 | socket_file = arg; | ||
| 701 | |||
| 702 | return NULL; | ||
| 703 | } | ||
| 650 | 704 | ||
| 651 | const char *add_referer_mysql_ignore(cmd_parms *parms, void *dummy, char *arg) | 705 | const char *add_referer_mysql_ignore(cmd_parms *parms, void *dummy, char *arg) |
| 652 | { | 706 | { |
| 653 | char **addme; | 707 | char **addme; |
| 654 | log_mysql_state *cls = get_module_config(parms->server->module_config, | 708 | log_mysql_state *cls = get_module_config(parms->server->module_config, &mysql_log_module); |
| 655 | &mysql_log_module); | ||
| 656 | 709 | ||
| 657 | addme = push_array(cls->referer_ignore_list); | 710 | addme = push_array(cls->referer_ignore_list); |
| 658 | *addme = pstrdup(cls->referer_ignore_list->pool, arg); | 711 | *addme = pstrdup(cls->referer_ignore_list->pool, arg); |
| @@ -662,8 +715,7 @@ const char *add_referer_mysql_ignore(cmd_parms *parms, void *dummy, char *arg) | |||
| 662 | const char *add_transfer_mysql_ignore(cmd_parms *parms, void *dummy, char *arg) | 715 | const char *add_transfer_mysql_ignore(cmd_parms *parms, void *dummy, char *arg) |
| 663 | { | 716 | { |
| 664 | char **addme; | 717 | char **addme; |
| 665 | log_mysql_state *cls = get_module_config(parms->server->module_config, | 718 | log_mysql_state *cls = get_module_config(parms->server->module_config, &mysql_log_module); |
| 666 | &mysql_log_module); | ||
| 667 | 719 | ||
| 668 | addme = push_array(cls->transfer_ignore_list); | 720 | addme = push_array(cls->transfer_ignore_list); |
| 669 | *addme = pstrdup(cls->transfer_ignore_list->pool, arg); | 721 | *addme = pstrdup(cls->transfer_ignore_list->pool, arg); |
| @@ -673,8 +725,7 @@ const char *add_transfer_mysql_ignore(cmd_parms *parms, void *dummy, char *arg) | |||
| 673 | const char *add_remhost_mysql_ignore(cmd_parms *parms, void *dummy, char *arg) | 725 | const char *add_remhost_mysql_ignore(cmd_parms *parms, void *dummy, char *arg) |
| 674 | { | 726 | { |
| 675 | char **addme; | 727 | char **addme; |
| 676 | log_mysql_state *cls = get_module_config(parms->server->module_config, | 728 | log_mysql_state *cls = get_module_config(parms->server->module_config, &mysql_log_module); |
| 677 | &mysql_log_module); | ||
| 678 | 729 | ||
| 679 | addme = push_array(cls->remhost_ignore_list); | 730 | addme = push_array(cls->remhost_ignore_list); |
| 680 | *addme = pstrdup(cls->remhost_ignore_list->pool, arg); | 731 | *addme = pstrdup(cls->remhost_ignore_list->pool, arg); |
| @@ -682,24 +733,91 @@ const char *add_remhost_mysql_ignore(cmd_parms *parms, void *dummy, char *arg) | |||
| 682 | } | 733 | } |
| 683 | 734 | ||
| 684 | 735 | ||
| 685 | /* | 736 | |
| 686 | * Apache-specific hooks into the module code | 737 | |
| 687 | * that are defined in the array 'mysql_lgog_module' (at EOF) | 738 | /*------------------------------------------------------------* |
| 739 | * Apache-specific hooks into the module code * | ||
| 740 | * that are defined in the array 'mysql_lgog_module' (at EOF) * | ||
| 741 | *------------------------------------------------------------*/ | ||
| 742 | |||
| 743 | |||
| 744 | /* | ||
| 745 | * This function is called during server initialisation when an heavy-weight | ||
| 746 | * process (such as a child) is being initialised. As with the | ||
| 747 | * module-initialisation function, any information that needs to be recorded | ||
| 748 | * must be in static cells, since there's no configuration record. | ||
| 749 | * | ||
| 750 | * There is no return value. | ||
| 688 | */ | 751 | */ |
| 752 | static void log_mysql_child_init(server_rec *s, pool *p) | ||
| 753 | { | ||
| 754 | int retval; | ||
| 755 | |||
| 756 | retval = open_logdb_link(); | ||
| 757 | #ifdef DEBUG | ||
| 758 | if (retval > 0) { | ||
| 759 | ap_log_error(APLOG_MARK,DEBUGLEVEL,s,"open_logdb_link successful"); | ||
| 760 | } | ||
| 761 | #endif | ||
| 762 | } | ||
| 689 | 763 | ||
| 764 | /* | ||
| 765 | * This function is called when an heavy-weight process (such as a child) is | ||
| 766 | * being run down or destroyed. As with the child-initialisation function, | ||
| 767 | * any information that needs to be recorded must be in static cells, since | ||
| 768 | * there's no configuration record. | ||
| 769 | * | ||
| 770 | * There is no return value. | ||
| 771 | */ | ||
| 772 | static void log_mysql_child_exit(server_rec *s, pool *p) | ||
| 773 | { | ||
| 774 | mysql_close(mysql_log); | ||
| 775 | } | ||
| 776 | |||
| 777 | |||
| 778 | /* | ||
| 779 | void *log_mysql_initializer(server_rec *main_server, pool *p) | ||
| 780 | { | ||
| 781 | server_rec *s; | ||
| 782 | |||
| 783 | log_mysql_state main_conf = ap_get_module_config(main_server->module_config, &mysql_log_module); | ||
| 690 | 784 | ||
| 691 | /* Set up space for the various major configuration options */ | 785 | for (server_rec *s = main_server; s; s = s->next) { |
| 786 | conf = ap_get_module_config(s->module_config, &mysql_log_module); | ||
| 787 | if (conf->transfer_log_format == NULL && s != main_server) { | ||
| 788 | *conf = *main_conf; | ||
| 789 | } | ||
| 790 | |||
| 791 | } | ||
| 792 | */ | ||
| 793 | |||
| 794 | /* | ||
| 795 | * This function gets called to create a per-server configuration | ||
| 796 | * record. It will always be called for the main server and | ||
| 797 | * for each virtual server that is established. Each server maintains | ||
| 798 | * its own state that is separate from the others' states. | ||
| 799 | * | ||
| 800 | * The return value is a pointer to the created module-specific | ||
| 801 | * structure. | ||
| 802 | */ | ||
| 692 | void *log_mysql_make_state(pool *p, server_rec *s) | 803 | void *log_mysql_make_state(pool *p, server_rec *s) |
| 693 | { | 804 | { |
| 694 | log_mysql_state *cls = (log_mysql_state *) palloc(p, sizeof(log_mysql_state)); | 805 | |
| 806 | log_mysql_state *cls = (log_mysql_state *) ap_palloc(p, sizeof(log_mysql_state)); | ||
| 807 | |||
| 695 | 808 | ||
| 696 | cls->referer_table_name = cls->agent_table_name = cls->transfer_table_name = ""; | 809 | cls->transfer_table_name = NULL; |
| 810 | cls->transfer_log_format = NULL; | ||
| 697 | 811 | ||
| 698 | cls->referer_ignore_list = make_array(p, 1, sizeof(char *)); | 812 | cls->referer_ignore_list = make_array(p, 1, sizeof(char *)); |
| 699 | cls->transfer_ignore_list = make_array(p, 1, sizeof(char *)); | 813 | cls->transfer_ignore_list = make_array(p, 1, sizeof(char *)); |
| 700 | cls->remhost_ignore_list = make_array(p, 1, sizeof(char *)); | 814 | cls->remhost_ignore_list = make_array(p, 1, sizeof(char *)); |
| 701 | 815 | ||
| 702 | cls->transfer_log_format = ""; | 816 | cls->table_made = 0; |
| 817 | cls->create_tables = 0; | ||
| 818 | |||
| 819 | cls->preserve_file = "/tmp/mysql-preserve"; | ||
| 820 | |||
| 703 | return (void *) cls; | 821 | return (void *) cls; |
| 704 | } | 822 | } |
| 705 | 823 | ||
| @@ -708,12 +826,6 @@ void *log_mysql_make_state(pool *p, server_rec *s) | |||
| 708 | * Structure: command, function called, NULL, where available, how many arguments, verbose description | 826 | * Structure: command, function called, NULL, where available, how many arguments, verbose description |
| 709 | */ | 827 | */ |
| 710 | command_rec log_mysql_cmds[] = { | 828 | command_rec log_mysql_cmds[] = { |
| 711 | {"MySQLRefererLogTable", set_referer_log_mysql_table, NULL, RSRC_CONF, TAKE1, | ||
| 712 | "The MySQL table that holds the referer log"} | ||
| 713 | , | ||
| 714 | {"MySQLAgentLogTable", set_agent_log_mysql_table, NULL, RSRC_CONF, TAKE1, | ||
| 715 | "The MySQL table that holds the agent log"} | ||
| 716 | , | ||
| 717 | {"MySQLTransferLogTable", set_transfer_log_mysql_table, NULL, RSRC_CONF, TAKE1, | 829 | {"MySQLTransferLogTable", set_transfer_log_mysql_table, NULL, RSRC_CONF, TAKE1, |
| 718 | "The MySQL table that holds the transfer log"} | 830 | "The MySQL table that holds the transfer log"} |
| 719 | , | 831 | , |
| @@ -738,6 +850,18 @@ command_rec log_mysql_cmds[] = { | |||
| 738 | {"MySQLLoginInfo", set_log_mysql_info, NULL, RSRC_CONF, TAKE3, | 850 | {"MySQLLoginInfo", set_log_mysql_info, NULL, RSRC_CONF, TAKE3, |
| 739 | "The MySQL host, user-id and password for logging"} | 851 | "The MySQL host, user-id and password for logging"} |
| 740 | , | 852 | , |
| 853 | {"MySQLCreateTables", set_log_mysql_create, NULL, RSRC_CONF, FLAG, | ||
| 854 | "Turn on module's capability to create its SQL tables on the fly"} | ||
| 855 | , | ||
| 856 | {"MySQLMassVirtualHosting", set_massvirtual, NULL, RSRC_CONF, FLAG, | ||
| 857 | "Activates option(s) useful for ISPs performing mass virutal hosting"} | ||
| 858 | , | ||
| 859 | {"MySQLPreserveFile", set_log_mysql_preserve_file, NULL, RSRC_CONF, TAKE1, | ||
| 860 | "Name of the file to use for data preservation during database downtime"} | ||
| 861 | , | ||
| 862 | {"MySQLSocketFile", set_mysql_socket_file, NULL, RSRC_CONF, TAKE1, | ||
| 863 | "Name of the file to employ for socket connections to MySQL"} | ||
| 864 | , | ||
| 741 | {NULL} | 865 | {NULL} |
| 742 | }; | 866 | }; |
| 743 | 867 | ||
| @@ -749,102 +873,25 @@ command_rec log_mysql_cmds[] = { | |||
| 749 | int log_mysql_transaction(request_rec *orig) | 873 | int log_mysql_transaction(request_rec *orig) |
| 750 | { | 874 | { |
| 751 | char **ptrptr, **ptrptr2; | 875 | char **ptrptr, **ptrptr2; |
| 752 | log_mysql_state *cls = get_module_config(orig->server->module_config, | 876 | log_mysql_state *cls = get_module_config(orig->server->module_config, &mysql_log_module); |
| 753 | &mysql_log_module); | 877 | const char *str; |
| 754 | char *str; | ||
| 755 | const char *referer; | ||
| 756 | request_rec *r; | 878 | request_rec *r; |
| 757 | int retvalue = DECLINED; | ||
| 758 | int referer_needed, agent_needed, transfer_needed; | ||
| 759 | |||
| 760 | 879 | ||
| 761 | /* Are there configuration directives for these SQL logs? For each found | 880 | /* Are there configuration directives for these SQL logs? For each found |
| 762 | * config directive that is found, mark that type as 'needed'. | 881 | * config directive that is found, mark that type as 'needed'. |
| 763 | */ | 882 | */ |
| 764 | referer_needed = ((cls->referer_table_name[0] != '\0') ? 1 : 0); | 883 | if ( ((cls->transfer_table_name == NULL) ? 1 : 0) ) { |
| 765 | agent_needed = ((cls->agent_table_name[0] != '\0') ? 1 : 0); | 884 | return DECLINED; |
| 766 | transfer_needed = ((cls->transfer_table_name[0] != '\0') ? 1 : 0); | 885 | } else { |
| 767 | |||
| 768 | if (!referer_needed && !agent_needed && !transfer_needed) { | ||
| 769 | return OK; | ||
| 770 | } | ||
| 771 | |||
| 772 | |||
| 773 | for (r = orig; r->next; r = r->next) { | ||
| 774 | continue; | ||
| 775 | } | ||
| 776 | |||
| 777 | /* Log the 'referer' to its own log if configured to do so. */ | ||
| 778 | if (referer_needed) { | ||
| 779 | retvalue = OK; | ||
| 780 | referer = table_get(orig->headers_in, "Referer"); | ||
| 781 | if (referer != NULL) { | ||
| 782 | |||
| 783 | /* The following is an upsetting mess of pointers, I'm sorry | ||
| 784 | * Anyone with the motiviation and/or the time should feel free | ||
| 785 | * to make this cleaner... */ | ||
| 786 | ptrptr2 = (char **) (cls->referer_ignore_list->elts + (cls->referer_ignore_list->nelts * cls->referer_ignore_list->elt_size)); | ||
| 787 | |||
| 788 | /* Go through each element of the ignore list and compare it to the | ||
| 789 | * referer_host. If we get a match, return without logging */ | ||
| 790 | for (ptrptr = (char **) cls->referer_ignore_list->elts; ptrptr < ptrptr2; ptrptr = (char **) ((char *) ptrptr + cls->referer_ignore_list->elt_size)) { | ||
| 791 | if (strstr(referer, *ptrptr)) { | ||
| 792 | return OK; | ||
| 793 | } | ||
| 794 | } | ||
| 795 | str = pstrcat(orig->pool, "insert into ", cls->referer_table_name, " (referer,url,time_stamp) values ('", mysql_escape_log(referer, orig->pool), "','", mysql_escape_log(r->uri, orig->pool), "',unix_timestamp(now()) )", NULL); | ||
| 796 | |||
| 797 | if (mysql_log == NULL) { /* child's mysql link not up, re-establish it */ | ||
| 798 | open_logdb_link(); | ||
| 799 | if (mysql_log == NULL) { | ||
| 800 | preserve_entry(r, str); | ||
| 801 | return OK; | ||
| 802 | } else { | ||
| 803 | ap_log_error(APLOG_MARK,NOTICELEVEL,orig->server,"MySQL: httpd child established database connection"); | ||
| 804 | safe_mysql_query(orig, str); | ||
| 805 | } | ||
| 806 | } else { | ||
| 807 | safe_mysql_query(orig, str); | ||
| 808 | } | ||
| 809 | |||
| 810 | } | ||
| 811 | } | ||
| 812 | |||
| 813 | /* Log the 'user agent' to its own log if configured to do so. */ | ||
| 814 | if (agent_needed) { | ||
| 815 | const char *agent, *str; | ||
| 816 | |||
| 817 | retvalue = OK; | ||
| 818 | agent = table_get(orig->headers_in, "User-Agent"); | ||
| 819 | |||
| 820 | if (agent != NULL) { | ||
| 821 | str = pstrcat(orig->pool, "insert into ", cls->agent_table_name, "(agent,time_stamp) values ('", mysql_escape_log(agent, orig->pool), "',unix_timestamp(now()) )", NULL); | ||
| 822 | |||
| 823 | if (mysql_log == NULL) { /* child's mysql link not up, re-establish it */ | ||
| 824 | open_logdb_link(); | ||
| 825 | if (mysql_log == NULL) { | ||
| 826 | preserve_entry(r, str); | ||
| 827 | return OK; | ||
| 828 | } else { | ||
| 829 | ap_log_error(APLOG_MARK,NOTICELEVEL,orig->server,"MySQL: httpd child established database connection"); | ||
| 830 | safe_mysql_query(orig, str); | ||
| 831 | } | ||
| 832 | } else { | ||
| 833 | safe_mysql_query(orig, str); | ||
| 834 | } | ||
| 835 | } | ||
| 836 | } | ||
| 837 | |||
| 838 | /* Log the transfer to its own log if configured to do so. */ | ||
| 839 | if (transfer_needed) { | ||
| 840 | const char *thehost; | 886 | const char *thehost; |
| 841 | |||
| 842 | char *fields = "", *values = ""; | 887 | char *fields = "", *values = ""; |
| 843 | const char *formatted_item; | 888 | const char *formatted_item; |
| 844 | int i, j, length; | 889 | int i, j, length; |
| 890 | char *createstring = NULL; | ||
| 845 | 891 | ||
| 846 | retvalue = OK; | 892 | for (r = orig; r->next; r = r->next) { |
| 847 | 893 | continue; | |
| 894 | } | ||
| 848 | 895 | ||
| 849 | /* The following is a stolen upsetting mess of pointers, I'm sorry | 896 | /* The following is a stolen upsetting mess of pointers, I'm sorry |
| 850 | * Anyone with the motiviation and/or the time should feel free | 897 | * Anyone with the motiviation and/or the time should feel free |
| @@ -856,7 +903,7 @@ int log_mysql_transaction(request_rec *orig) | |||
| 856 | if (r->uri) { | 903 | if (r->uri) { |
| 857 | for (ptrptr = (char **) cls->transfer_ignore_list->elts; ptrptr < ptrptr2; ptrptr = (char **) ((char *) ptrptr + cls->transfer_ignore_list->elt_size)) { | 904 | for (ptrptr = (char **) cls->transfer_ignore_list->elts; ptrptr < ptrptr2; ptrptr = (char **) ((char *) ptrptr + cls->transfer_ignore_list->elt_size)) { |
| 858 | if (strstr(r->uri, *ptrptr)) { | 905 | if (strstr(r->uri, *ptrptr)) { |
| 859 | return retvalue; | 906 | return OK; |
| 860 | } | 907 | } |
| 861 | } | 908 | } |
| 862 | } | 909 | } |
| @@ -868,13 +915,13 @@ int log_mysql_transaction(request_rec *orig) | |||
| 868 | if (thehost) { | 915 | if (thehost) { |
| 869 | for (ptrptr = (char **) cls->remhost_ignore_list->elts; ptrptr < ptrptr2; ptrptr = (char **) ((char *) ptrptr + cls->remhost_ignore_list->elt_size)) { | 916 | for (ptrptr = (char **) cls->remhost_ignore_list->elts; ptrptr < ptrptr2; ptrptr = (char **) ((char *) ptrptr + cls->remhost_ignore_list->elt_size)) { |
| 870 | if (strstr(thehost, *ptrptr)) { | 917 | if (strstr(thehost, *ptrptr)) { |
| 871 | return retvalue; | 918 | return OK; |
| 872 | } | 919 | } |
| 873 | } | 920 | } |
| 874 | } | 921 | } |
| 875 | 922 | ||
| 876 | /* If not specified by the user, use the default format */ | 923 | /* If not specified by the user, use the default format */ |
| 877 | if (cls->transfer_log_format[0] == '\0') { | 924 | if (cls->transfer_log_format == NULL) { |
| 878 | cls->transfer_log_format = "AbHhmRSsTUuv"; | 925 | cls->transfer_log_format = "AbHhmRSsTUuv"; |
| 879 | } | 926 | } |
| 880 | length = strlen(cls->transfer_log_format); | 927 | length = strlen(cls->transfer_log_format); |
| @@ -883,8 +930,10 @@ int log_mysql_transaction(request_rec *orig) | |||
| 883 | * what the user has configured. */ | 930 | * what the user has configured. */ |
| 884 | for (i = 0; i < length; i++) { | 931 | for (i = 0; i < length; i++) { |
| 885 | j = 0; | 932 | j = 0; |
| 933 | |||
| 886 | while (log_mysql_item_keys[j].ch) { | 934 | while (log_mysql_item_keys[j].ch) { |
| 887 | if (log_mysql_item_keys[j].ch == cls->transfer_log_format[i]) { | 935 | |
| 936 | if (log_mysql_item_keys[j].ch == cls->transfer_log_format[i]) { | ||
| 888 | /* Yes, this key is one of the configured keys. | 937 | /* Yes, this key is one of the configured keys. |
| 889 | * Call the key's function and put the returned value into 'formatted_item' */ | 938 | * Call the key's function and put the returned value into 'formatted_item' */ |
| 890 | formatted_item = log_mysql_item_keys[j].func(log_mysql_item_keys[j].want_orig_default ? orig : r, ""); | 939 | formatted_item = log_mysql_item_keys[j].func(log_mysql_item_keys[j].want_orig_default ? orig : r, ""); |
| @@ -894,91 +943,134 @@ int log_mysql_transaction(request_rec *orig) | |||
| 894 | formatted_item = ""; | 943 | formatted_item = ""; |
| 895 | } else if (formatted_item[0] == '-' && formatted_item[1] == '\0' && !log_mysql_item_keys[j].string_contents) { | 944 | } else if (formatted_item[0] == '-' && formatted_item[1] == '\0' && !log_mysql_item_keys[j].string_contents) { |
| 896 | /* If apache tried to log a '-' character for a numeric field, convert that to a zero | 945 | /* If apache tried to log a '-' character for a numeric field, convert that to a zero |
| 897 | * because the database expects an integer. */ | 946 | * because the database expects a numeral and will reject the '-' character. */ |
| 898 | formatted_item = "0"; | 947 | formatted_item = "0"; |
| 899 | } | 948 | } |
| 900 | 949 | ||
| 901 | /* Append the fieldname and value-to-insert to teh appropriate strings, quoting stringvals with ' as appropriate */ | 950 | /* Append the fieldname and value-to-insert to the appropriate strings, quoting stringvals with ' as appropriate */ |
| 902 | fields = pstrcat(orig->pool, fields, (i > 0 ? "," : ""), log_mysql_item_keys[j].sql_field_name, NULL); | 951 | fields = pstrcat(r->pool, fields, (i > 0 ? "," : ""), |
| 903 | values = pstrcat(orig->pool, values, (i > 0 ? "," : ""), (log_mysql_item_keys[j].string_contents ? "'" : ""), mysql_escape_log(formatted_item, orig->pool), (log_mysql_item_keys[j].string_contents ? "'" : ""), NULL); | 952 | log_mysql_item_keys[j].sql_field_name, NULL); |
| 953 | |||
| 954 | values = pstrcat(r->pool, values, (i > 0 ? "," : ""), | ||
| 955 | (log_mysql_item_keys[j].string_contents ? "'" : ""), | ||
| 956 | escape_query(formatted_item, r->pool), | ||
| 957 | (log_mysql_item_keys[j].string_contents ? "'" : ""), NULL); | ||
| 904 | break; | 958 | break; |
| 905 | } | 959 | } |
| 906 | j++; | 960 | j++; |
| 961 | |||
| 907 | } | 962 | } |
| 908 | } | 963 | } |
| 909 | 964 | ||
| 910 | /* Set up the actual INSERT statement and execute it. */ | 965 | |
| 911 | str = pstrcat(orig->pool, "insert into ", cls->transfer_table_name, " (", fields, ") values (", values, ")", NULL); | 966 | /* Is this virtual server's table flagged as made? We flag it as such in order |
| 967 | * to avoid extra processing with each request. If it's not flagged as made, | ||
| 968 | * set up the CREATE string. | ||
| 969 | */ | ||
| 970 | if ((cls->table_made != 1) && (cls->create_tables != 0)) { | ||
| 971 | char *createprefix = "create table if not exists "; | ||
| 972 | char *createsuffix = | ||
| 973 | " (agent varchar(255),\ | ||
| 974 | bytes_sent int unsigned,\ | ||
| 975 | child_pid smallint unsigned,\ | ||
| 976 | cookie varchar(255),\ | ||
| 977 | request_file varchar(255),\ | ||
| 978 | referer varchar(255),\ | ||
| 979 | remote_host varchar(50),\ | ||
| 980 | remote_logname varchar(50),\ | ||
| 981 | remote_user varchar(50),\ | ||
| 982 | request_duration smallint unsigned,\ | ||
| 983 | request_line varchar(255),\ | ||
| 984 | request_method varchar(6),\ | ||
| 985 | request_protocol varchar(10),\ | ||
| 986 | request_time char(28),\ | ||
| 987 | request_uri varchar(50),\ | ||
| 988 | server_port smallint unsigned,\ | ||
| 989 | ssl_cipher varchar(25),\ | ||
| 990 | ssl_keysize smallint unsigned,\ | ||
| 991 | ssl_maxkeysize smallint unsigned,\ | ||
| 992 | status smallint unsigned,\ | ||
| 993 | time_stamp int unsigned,\ | ||
| 994 | virtual_host varchar(50))"; | ||
| 995 | |||
| 996 | /* Find memory long enough to hold the whole CREATE string + \0 */ | ||
| 997 | /* old way: | ||
| 998 | * createstring = (char*)ap_palloc(orig->pool,(strlen(createprefix) + strlen(cls->transfer_table_name) + strlen(createsuffix) + 1) * sizeof(char)); | ||
| 999 | * strcpy (createstring, createprefix); | ||
| 1000 | * strcat (createstring, cls->transfer_table_name); | ||
| 1001 | * strcat (createstring, createsuffix); */ | ||
| 1002 | |||
| 1003 | createstring = ap_pstrcat(orig->pool, createprefix, cls->transfer_table_name, createsuffix, NULL); | ||
| 1004 | |||
| 1005 | #ifdef DEBUG | ||
| 1006 | ap_log_error(APLOG_MARK,DEBUGLEVEL,orig->server,"create string: %s", createstring); | ||
| 1007 | #endif | ||
| 912 | 1008 | ||
| 1009 | } | ||
| 1010 | |||
| 1011 | /* Set up the actual INSERT statement and escape it. */ | ||
| 1012 | str = ap_pstrcat(r->pool, "insert into ", cls->transfer_table_name, " (", fields, ") values (", values, ")", NULL); | ||
| 913 | 1013 | ||
| 1014 | #ifdef DEBUG | ||
| 1015 | ap_log_error(APLOG_MARK,DEBUGLEVEL,r->server,"insert string: %s", str); | ||
| 1016 | #endif | ||
| 1017 | |||
| 1018 | |||
| 914 | /* How's our mysql link integrity? */ | 1019 | /* How's our mysql link integrity? */ |
| 915 | if (mysql_log == NULL) { | 1020 | if (mysql_log == NULL) { |
| 916 | 1021 | ||
| 917 | /* Try to regain the link */ | 1022 | /* Make a try to establish the link */ |
| 918 | open_logdb_link(); | 1023 | open_logdb_link(); |
| 919 | 1024 | ||
| 920 | if (mysql_log == NULL) { | 1025 | if (mysql_log == NULL) { |
| 921 | /* Unable to re-establish a DB link, so assume that it's really | 1026 | /* Unable to re-establish a DB link, so assume that it's really |
| 922 | * gone and send the entry to the preserve file instead. */ | 1027 | * gone and send the entry to the preserve file instead. |
| 923 | preserve_entry(r, str); | 1028 | * Note that we don't keep logging the db error over and over. */ |
| 1029 | preserve_entry(orig, str); | ||
| 924 | return OK; | 1030 | return OK; |
| 925 | } else { | 1031 | } else { |
| 926 | /* Whew, we got the DB link back */ | 1032 | /* Whew, we got the DB link back */ |
| 927 | ap_log_error(APLOG_MARK,NOTICELEVEL,orig->server,"MySQL: httpd child established database connection"); | 1033 | ap_log_error(APLOG_MARK,NOTICELEVEL,orig->server,"httpd child established database connection"); |
| 928 | safe_mysql_query(orig, str); | ||
| 929 | } | 1034 | } |
| 930 | } else { | ||
| 931 | /* Everything was fine */ | ||
| 932 | safe_mysql_query(orig, str); | ||
| 933 | } | 1035 | } |
| 1036 | |||
| 1037 | if ((cls->table_made != 1) && (cls->create_tables != 0)) { | ||
| 1038 | mysql_query(mysql_log,createstring); | ||
| 1039 | cls->table_made = 1; | ||
| 1040 | } | ||
| 1041 | |||
| 1042 | /* Make the insert */ | ||
| 1043 | safe_mysql_query(orig, str); | ||
| 934 | 1044 | ||
| 1045 | return OK; | ||
| 935 | } | 1046 | } |
| 936 | return retvalue; | ||
| 937 | } | 1047 | } |
| 938 | 1048 | ||
| 939 | 1049 | ||
| 940 | /* Called on the exit of an httpd child process */ | ||
| 941 | static void log_mysql_child_exit(server_rec *s, pool *p) | ||
| 942 | { | ||
| 943 | mysql_close(mysql_log); | ||
| 944 | } | ||
| 945 | |||
| 946 | /* Called on the init of an httpd child process */ | ||
| 947 | static void log_mysql_child_init(server_rec *s, pool *p) | ||
| 948 | { | ||
| 949 | int retval; | ||
| 950 | |||
| 951 | retval = open_logdb_link(); | ||
| 952 | #ifdef DEBUG | ||
| 953 | if (retval > 0) { | ||
| 954 | ap_log_error(APLOG_MARK,DEBUGLEVEL,s,"MySQL: open_logdb_link successful"); | ||
| 955 | } | ||
| 956 | #endif | ||
| 957 | } | ||
| 958 | 1050 | ||
| 959 | 1051 | ||
| 960 | /* The configuration array that sets up the hooks into the module. */ | 1052 | /* The configuration array that sets up the hooks into the module. */ |
| 961 | module mysql_log_module = { | 1053 | module mysql_log_module = { |
| 962 | STANDARD_MODULE_STUFF, | 1054 | STANDARD_MODULE_STUFF, |
| 963 | NULL, /* initializer */ | 1055 | NULL, /* module initializer */ |
| 964 | NULL, /* create per-dir config */ | 1056 | NULL, /* create per-dir config */ |
| 965 | NULL, /* merge per-dir config */ | 1057 | NULL, /* merge per-dir config */ |
| 966 | log_mysql_make_state, /* server config */ | 1058 | log_mysql_make_state, /* create server config */ |
| 967 | NULL, /* merge server config */ | 1059 | NULL, /* merge server config */ |
| 968 | log_mysql_cmds, /* command table */ | 1060 | log_mysql_cmds, /* config directive table */ |
| 969 | NULL, /* handlers */ | 1061 | NULL, /* [9] content handlers */ |
| 970 | NULL, /* filename translation */ | 1062 | NULL, /* [2] URI-to-filename translation */ |
| 971 | NULL, /* check_user_id */ | 1063 | NULL, /* [5] check/validate user_id */ |
| 972 | NULL, /* check auth */ | 1064 | NULL, /* [6] check authorization */ |
| 973 | NULL, /* check access */ | 1065 | NULL, /* [4] check access by host */ |
| 974 | NULL, /* type_checker */ | 1066 | NULL, /* [7] MIME type checker/setter */ |
| 975 | NULL, /* fixups */ | 1067 | NULL, /* [8] fixups */ |
| 976 | log_mysql_transaction, /* logger */ | 1068 | log_mysql_transaction, /* [10] logger */ |
| 977 | NULL, /* header parser */ | 1069 | NULL /* [3] header parser */ |
| 978 | #if MODULE_MAGIC_NUMBER >= 19970728 /* 1.3-dev or later support these additionals... */ | 1070 | #if MODULE_MAGIC_NUMBER >= 19970728 /* 1.3-dev or later support these additionals... */ |
| 979 | log_mysql_child_init, /* child_init */ | 1071 | ,log_mysql_child_init, /* child process initializer */ |
| 980 | log_mysql_child_exit, /* process exit/cleanup */ | 1072 | log_mysql_child_exit, /* process exit/cleanup */ |
| 981 | NULL /* [#0] post read-request */ | 1073 | NULL /* [1] post read-request */ |
| 982 | #endif | 1074 | #endif |
| 983 | 1075 | ||
| 984 | }; | 1076 | }; |
