/** @file
Parser: apache 1.3 and 2.2 module
Copyright (c) 2001-2026 Art. Lebedev Studio (https://www.artlebedev.com)
Authors: Konstantin Morshnev <moko@design.ru>, Alexandr Petrosian <paf@design.ru>
*/
#include "httpd.h"
#include "http_config.h"
#include "http_core.h"
#include "http_log.h"
#include "http_main.h"
#include "http_protocol.h"
#include "util_script.h"
#include "pa_httpd.h"
volatile const char * IDENT_MOD_PARSER3_C="$Id: mod_parser3.c,v 1.26 2026/04/25 13:38:46 moko Exp $" IDENT_PA_HTTPD_H;
#define PARSER3_HANDLER "parser3-handler"
/*
* To Ease Compatibility
*/
#ifdef STANDARD20_MODULE_STUFF
#include "apr_strings.h"
#include "ap_mpm.h"
#define ap_pcalloc apr_pcalloc
#define ap_pstrdup apr_pstrdup
#define ap_table_get apr_table_get
#define ap_table_elts apr_table_elts
#define ap_table_addn apr_table_addn
#define ap_table_do apr_table_do
#else
#include "ap_alloc.h"
#define apr_pool_t pool
#define apr_table_t table
#endif /* STANDARD20_MODULE_STUFF */
/*
* Declare ourselves so the configuration routines can find and know us.
* We'll fill it in at the end of the module.
*/
#ifdef STANDARD20_MODULE_STUFF
module AP_MODULE_DECLARE_DATA parser3_module;
static int is_threaded = 0;
#else
module MODULE_VAR_EXPORT parser3_module;
#endif
/*
* Locate our directory configuration record for the current request.
*/
static Parser_module_config *our_dconfig(request_rec *r) {
return (Parser_module_config *) ap_get_module_config(r->per_dir_config, &parser3_module);
}
static const char* cmd_parser_config(cmd_parms *cmd, void *mconfig, const char *file_spec) {
Parser_module_config *cfg = (Parser_module_config *) mconfig;
cfg->parser_config_filespec=file_spec;
return NULL;
}
/*
* Now let's declare routines for each of the callback phase in order.
*/
static int parser_handler(request_rec *r) {
#ifdef STANDARD20_MODULE_STUFF
if(strcmp(r->handler, PARSER3_HANDLER))
return DECLINED;
if(is_threaded){
const char *message="Parser3 module requires 'a2enmod mpm_prefork'";
r->status=HTTP_INTERNAL_SERVER_ERROR;
r->content_type="text/plain";
ap_rwrite(message, strlen(message), r);
return OK;
}
#endif
// we setup module here to avoid GPF on init with php5-xsl installed
pa_setup_module_cells();
{
// converting to parser version
pa_request_rec pr={
r,
r->pool,
r->header_only,
&r->status,
r->method,
r->headers_out,
r->subprocess_env,
&r->content_type,
r->uri,
r->filename,
r->path_info,
r->args,
#ifdef STANDARD20_MODULE_STUFF
r->finfo.filetype == 0
#else
r->finfo.st_mode == 0
#endif
};
return pa_parser_handler(&pr, our_dconfig(r));
}
}
/*
* This function is called during process initialisation.
*/
#ifdef STANDARD20_MODULE_STUFF
static void parser_child_init(apr_pool_t *p, server_rec *s) {
#else
static void parser_module_init(server_rec *s, apr_pool_t *p) {
#endif
// ap_log_perror(APLOG_MARK, APLOG_EMERG, 0, p, "parser inited %d", getpid());
}
/*
* All our process-death routine does is add its trace to the log.
*/
static void parser_module_done(server_rec *s, apr_pool_t *p) {
pa_destroy_module_cells();
}
/*
* This function gets called to create a per-directory configuration record.
*/
static void *parser_create_dir_config(apr_pool_t *p, char *dirspec) {
Parser_module_config *cfg= ap_pcalloc(p, sizeof(Parser_module_config));
cfg->parser_config_filespec=NULL;
return cfg;
}
/*
* This function gets called to create a per-server configuration record.
*/
static void *parser_create_server_config(apr_pool_t *p, server_rec *s) {
Parser_module_config *cfg= ap_pcalloc(p, sizeof(Parser_module_config));
cfg->parser_config_filespec=NULL;
return cfg;
}
/*
* List of directives specific to our module.
*/
static const command_rec parser_cmds[] =
{
{"ParserConfig", (const char *(*)(cmd_parms *, void *))cmd_parser_config, 0, OR_OPTIONS, TAKE1, "Parser config filespec"},
{NULL}
};
/*--------------------------------------------------------------------------*/
/* Now the list of content handlers available from this module. */
/*--------------------------------------------------------------------------*/
#ifndef STANDARD20_MODULE_STUFF
static const handler_rec parser_handlers[] =
{
{PARSER3_HANDLER, parser_handler},
{NULL}
};
#endif
/*--------------------------------------------------------------------------*/
/* Finally, the list of callback routines and data structures that */
/* provide the hooks into our module from the other parts of the server. */
/*--------------------------------------------------------------------------*/
#ifdef STANDARD20_MODULE_STUFF
/*
* register hooks.
*/
static void parser_register_hooks(apr_pool_t* pool)
{
int mpm_query_info;
is_threaded = ap_mpm_query(AP_MPMQ_IS_THREADED, &mpm_query_info) == APR_SUCCESS && mpm_query_info != AP_MPMQ_NOT_SUPPORTED;
ap_hook_handler(parser_handler, NULL, NULL, APR_HOOK_MIDDLE);
ap_hook_child_init(parser_child_init, NULL, NULL, APR_HOOK_MIDDLE);
};
module AP_MODULE_DECLARE_DATA parser3_module =
{
STANDARD20_MODULE_STUFF,
#else
module MODULE_VAR_EXPORT parser3_module =
{
STANDARD_MODULE_STUFF,
parser_module_init, /* module initializer */
#endif
parser_create_dir_config, /* per-directory config creator */
0, /* dir config merger */
parser_create_server_config, /* server config creator */
0, /* server config merger */
parser_cmds, /* command apr_table_t */
#ifdef STANDARD20_MODULE_STUFF
parser_register_hooks /* register hooks */
#else
parser_handlers, /* [9] list of handlers */
0, /* [2] filename-to-URI translation */
0, /* [5] check/validate user_id */
0, /* [6] check user_id is valid *here* */
0, /* [4] check access by host address */
0, /* [7] MIME type checker/setter */
0, /* [8] fixups */
0, /* [10] logger */
0, /* [3] header parser */
0, /* process initializer */
parser_module_done /* process exit/cleanup */
#endif // STANDARD20_MODULE_STUFF
};
#if defined(_MSC_VER)
# define APACHE_WIN32_SRC "../../../../win32/apache22/"
# ifdef _DEBUG
# pragma comment(lib, APACHE_WIN32_SRC "srclib/apr/Debug/libapr-1.lib")
# pragma comment(lib, APACHE_WIN32_SRC "Debug/libhttpd.lib")
# else
# pragma comment(lib, APACHE_WIN32_SRC "srclib/apr/Release/libapr-1.lib")
# pragma comment(lib, APACHE_WIN32_SRC "Release/libhttpd.lib")
# endif
#endif
// interface to C++
void pa_ap_log_rerror(const char *file, int line, int level, const pa_request_rec *s, const char *fmt, ...) {
const char* str;
va_list l;
va_start(l, fmt);
str=va_arg(l, const char*);
va_end(l);
ap_log_rerror(file, line,
#if (AP_SERVER_MAJORVERSION_NUMBER == 2) && (AP_SERVER_MINORVERSION_NUMBER >= 4)
APLOG_MODULE_INDEX,
#endif
level,
#ifdef STANDARD20_MODULE_STUFF
0,
#endif
(request_rec*)s->real_request_rec, "%s", str);
}
void pa_ap_log_error(const char *file, int line, int level, const pa_server_rec *s, const char *fmt, ...) {
const char* str;
va_list l;
va_start(l, fmt);
str=va_arg(l, const char*);
va_end(l);
ap_log_error(file, line,
#if (AP_SERVER_MAJORVERSION_NUMBER == 2) && (AP_SERVER_MINORVERSION_NUMBER >= 4)
APLOG_MODULE_INDEX,
#endif
level,
#ifdef STANDARD20_MODULE_STUFF
0,
#endif
(server_rec*)s, "%s", str);
}
// ap_alloc.h
const char* pa_ap_table_get(const pa_table *t, const char *name) {
return ap_table_get((const apr_table_t*)t, name);
}
void pa_ap_table_addn(pa_table *t, const char *name, const char *val) {
ap_table_addn((apr_table_t*)t, name, val);
}
int pa_ap_table_size(const pa_table *t) {
return ap_table_elts((const apr_table_t*)t)->nelts;
}
void pa_ap_table_do(int (*comp) (void *, const char *, const char *), void *rec, const pa_table *t, ...) {
ap_table_do(comp, rec, (apr_table_t*)t, NULL);
}
char * pa_ap_pstrdup(pa_pool *p, const char *s) {
return ap_pstrdup((apr_pool_t*)p, s);
}
// http_protocol.h
int pa_ap_setup_client_block(pa_request_rec *r, int read_policy) {
return ap_setup_client_block((request_rec*)r->real_request_rec,
read_policy);
}
int pa_ap_should_client_block(pa_request_rec *r) {
return ap_should_client_block((request_rec*)r->real_request_rec);
}
long pa_ap_get_client_block(pa_request_rec *r, char *buffer, int bufsiz) {
return ap_get_client_block((request_rec*)r->real_request_rec,
buffer, bufsiz);
}
void pa_ap_send_http_header(pa_request_rec *r) {
// Apache2 send headers before body automatically
#ifndef STANDARD20_MODULE_STUFF
ap_send_http_header((request_rec*)r->real_request_rec);
#endif
}
int pa_ap_rwrite(const void *buf, int nbyte, pa_request_rec *r) {
return ap_rwrite(buf, nbyte, (request_rec*)r->real_request_rec);
}
// http_main.h
void pa_ap_hard_timeout(const char *s, pa_request_rec *r) {
// Apache 2 uses non-blocking I/O
#ifndef STANDARD20_MODULE_STUFF
ap_hard_timeout((char *)s, (request_rec*)r->real_request_rec);
#endif
}
void pa_ap_reset_timeout(pa_request_rec *r) {
#ifndef STANDARD20_MODULE_STUFF
ap_reset_timeout((request_rec*)r->real_request_rec);
#endif
}
void pa_ap_kill_timeout(pa_request_rec *r) {
#ifndef STANDARD20_MODULE_STUFF
ap_kill_timeout((request_rec*)r->real_request_rec);
#endif
}
// util_script.h
void pa_ap_add_cgi_vars(pa_request_rec *r) {
ap_add_cgi_vars((request_rec*)r->real_request_rec);
}
void pa_ap_add_common_vars(pa_request_rec *r) {
ap_add_common_vars((request_rec*)r->real_request_rec);
}
// signal.h
void (*pa_signal (int sig, void (*disp)(int)))(int) {
#ifndef _MSC_VER
if(sig==PA_SIGPIPE && disp==PA_SIG_IGN)
return signal(SIGPIPE, SIG_IGN);
#endif
return 0;
}
E-mail: