mod_charset_lite.c revision e485880d90245d55fd2f6808fd81819a31feef89
/* ====================================================================
* The Apache Software License, Version 1.1
*
* Copyright (c) 2000 The Apache Software Foundation. All rights
* reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
*
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
*
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in
* distribution.
*
* 3. The end-user documentation included with the redistribution,
* if any, must include the following acknowledgment:
* "This product includes software developed by the
* Apache Software Foundation (http://www.apache.org/)."
* Alternately, this acknowledgment may appear in the software itself,
* if and wherever such third-party acknowledgments normally appear.
*
* 4. The names "Apache" and "Apache Software Foundation" must
* not be used to endorse or promote products derived from this
* software without prior written permission. For written
* permission, please contact apache@apache.org.
*
* 5. Products derived from this software may not be called "Apache",
* nor may "Apache" appear in their name, without prior written
* permission of the Apache Software Foundation.
*
* THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED
* WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
* OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
* DISCLAIMED. IN NO EVENT SHALL THE APACHE SOFTWARE FOUNDATION OR
* ITS CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF
* USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
* OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
* OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE.
* ====================================================================
*
* This software consists of voluntary contributions made by many
* individuals on behalf of the Apache Software Foundation. For more
* information on the Apache Software Foundation, please see
*
* Portions of this software are based upon public domain software
* originally written at the National Center for Supercomputing Applications,
* University of Illinois, Urbana-Champaign.
*/
/*
* simple hokey charset recoding configuration module
*
* See mod_ebcdic and mod_charset for more thought-out examples. This
* one is just so Jeff can learn how a module works and experiment with
* basic character set recoding configuration.
*
* !!!This is an extremely cheap ripoff of mod_charset.c from Russian Apache!!!
*/
#include "httpd.h"
#include "http_config.h"
#include "http_core.h"
#include "http_log.h"
#include "http_main.h"
#include "http_protocol.h"
#include "http_request.h"
#include "util_charset.h"
#include "ap_buckets.h"
#include "util_filter.h"
#ifndef APACHE_XLATE
#endif
* space is left in the translation buffer
*/
* two buckets
*/
typedef struct charset_dir_t {
means uninitialized */
const char *charset_source; /* source encoding */
const char *charset_default; /* how to ship on wire */
/* charset_filter_ctx_t is created for each filter instance; because the same
* filter code is used for translating in both directions, we need this context
* data to tell the filter which translation handle to use; it also can hold a
* character which was split between buckets
*/
typedef struct charset_filter_ctx_t {
/* charset_req_t is available via r->request_config if any translation is
* being performed
*/
typedef struct charset_req_t {
{
return apr_pcalloc(p,sizeof(charset_dir_t));
}
{
/* If it is defined in the current container, use it. Otherwise, use the one
* from the enclosing container.
*/
a->debug =
a->charset_default =
a->charset_source =
return a;
}
/* CharsetSourceEnc charset
*/
const char *name)
{
return NULL;
}
/* CharsetDefault charset
*/
const char *name)
{
return NULL;
}
/* CharsetDefault charset
*/
{
if (arg) {
}
else {
}
return NULL;
}
/* find_code_page() is a fixup hook that decides if translation should be
* enabled; if so, it sets up request data for use by the filter registration
* hook so that it knows what to do
*/
static int find_code_page(request_rec *r)
{
const char *mime_type;
if (debug) {
"Entering handler, URI: %s FILENAME: %s METHOD: %d ARGS: %s PATH_INFO: %s "
"MIMETYPE: %s FLAGS: %d SUBREQ: %s, REDIR: %s, PROXY: %s",
r->rrx ? 1 : 0,
}
/* If we don't have a full directory configuration, bail out.
*/
if (debug) {
"incomplete configuration: src %s, dst %s",
}
return DECLINED;
}
/* catch proxy requests */
/* mod_rewrite indicators */
/* If this is a subrequest, bail out. We don't want to be setting up
* translation just because something like mod_autoindex wants to find the
* mime type for directory objects.
* (I won't swear that there aren't cases where we need to process
* subrequests :) ).
*/
if (r->main) {
if (debug) {
"skipping subrequest");
}
return DECLINED;
}
/* If mime type isn't text or message, bail out.
*/
if (debug) {
"mime type is %s; no translation selected",
}
return DECLINED;
}
if (debug) {
"dc: %X charset_source: %s charset_default: %s",
(unsigned)dc,
}
/* Get storage for the request data and the output filter context.
* We rarely need the input filter context, so allocate that separately.
*/
sizeof(charset_req_t) +
sizeof(charset_filter_ctx_t));
if (rv != APR_SUCCESS) {
"can't open translation %s->%s",
return HTTP_INTERNAL_SERVER_ERROR;
}
switch (r->method_number) {
case M_PUT:
case M_POST:
/* Set up input translation. Note: A request body can be included
* with the OPTIONS method, but for now we don't set up translation
* of it.
*/
if (rv != APR_SUCCESS) {
"can't open translation %s->%s",
return HTTP_INTERNAL_SERVER_ERROR;
}
/* Can't delete this yet :( #ifdef OLD */
if (rv != APR_SUCCESS) {
"can't set content input translation");
return HTTP_INTERNAL_SERVER_ERROR;
}
/* #endif */
}
return DECLINED;
}
/* xlate_register_filter() is a filter hook which decides whether or not
* to insert a translation filter for the current request.
*/
static void xlate_register_filter(request_rec *r)
{
/* Hey... don't be so quick to use reqinfo->dc here; reqinfo may be NULL */
if (debug) {
"xlate_register_filter() - "
"dc: %X charset_source: %s charset_default: %s",
(unsigned)dc,
}
}
#ifdef NOT_YET
/* ap_add_filter(xxx, yyy, r); */
}
#endif
}
/* stuff that sucks that I know of:
*
* bucket handling:
* why create an eos bucket when we see it come down the stream? just send the one
* passed as input
*
* translation mechanics:
* we don't handle characters that straddle more than two buckets; an error
* will be generated
*
* we don't signal an error if we get EOS but we're in the middle of an input
* character
*/
/* send_downstream() is passed the translated data; it puts it in a single-
* bucket brigade and passes the brigade to the next filter
*/
{
}
{
}
{
}
}
}
{
}
/* input buffer: */
const char **cur_str,
/* output buffer: */
char **out_str,
{
/* Keep adding bytes from the input string to the saved string until we
* 1) finish the input char
* 2) get an error
* or 3) run out of bytes to add
*/
do {
++*cur_str;
--*cur_len;
*out_str,
out_len);
if (rv == APR_SUCCESS) {
}
return rv;
}
/* xlate_filter() handles arbirary conversions from one charset to another...
* translation is determined in the fixup hook (find_code_page), which is
* where the filter's context data is set up... the context data gives us
* the translation handle
*/
{
const char *cur_str;
char tmp[XLATE_BUF_SIZE];
int done;
if (debug) {
"xlate_filter() - "
"dc: %X charset_source: %s charset_default: %s",
(unsigned)dc,
}
done = 0;
cur_len = 0;
space_avail = sizeof(tmp);
while (!done) {
if (!cur_len) { /* no bytes left to process in the current bucket... */
if (consumed_bucket) {
}
if (!dptr ||
done = 1;
break;
}
}
/* Try to fill up our tmp buffer with translated data. */
/* Rats... we need to finish a partial character from the previous
* bucket.
*/
char *tmp_tmp;
&tmp_tmp, &space_avail);
}
else {
/* Update input ptr and len after consuming some bytes */
/* We need to safe the final byte(s) for next time; we can't
* convert it until we look at the next bucket.
*/
rv = APR_SUCCESS;
cur_len = 0;
}
}
if (rv != APR_SUCCESS) {
/* bad input byte; we can't continue */
done = 1;
"xlate_filter() - apr_xlate_conv_buffer() failed");
}
if (space_avail < XLATE_MIN_BUFF_LEFT) {
/* It is time to flush, as there is not enough space left in the
* current output buffer to bother with converting more data.
*/
if (rv == APR_SUCCESS) {
done = 1;
}
/* tmp is now empty */
space_avail = sizeof(tmp);
}
}
if (rv == APR_SUCCESS) {
}
}
if (rv == APR_SUCCESS) {
if (cur_len == AP_END_OF_BRIGADE) {
}
}
else {
"xlate_filter() - returning error");
}
return rv;
}
static const command_rec cmds[] =
{
AP_INIT_TAKE1("CharsetSourceEnc",
NULL,
"source (html,cgi,ssi) file charset"),
AP_INIT_TAKE1("CharsetDefault",
NULL,
"name of default charset"),
AP_INIT_FLAG("CharsetDebug",
NULL,
"mod_charset_lite debug flag"),
{NULL}
};
static void charset_register_hooks(void)
{
/* The first function just registers this module's register_filter
* hook. The other associates a global name with the filter defined
* by this module.
*/
}
{
NULL,
NULL,
cmds,
NULL,
};