Blame modules/debugging/mod_bucketeer.c

Packit 90a5c9
/* Licensed to the Apache Software Foundation (ASF) under one or more
Packit 90a5c9
 * contributor license agreements.  See the NOTICE file distributed with
Packit 90a5c9
 * this work for additional information regarding copyright ownership.
Packit 90a5c9
 * The ASF licenses this file to You under the Apache License, Version 2.0
Packit 90a5c9
 * (the "License"); you may not use this file except in compliance with
Packit 90a5c9
 * the License.  You may obtain a copy of the License at
Packit 90a5c9
 *
Packit 90a5c9
 *     http://www.apache.org/licenses/LICENSE-2.0
Packit 90a5c9
 *
Packit 90a5c9
 * Unless required by applicable law or agreed to in writing, software
Packit 90a5c9
 * distributed under the License is distributed on an "AS IS" BASIS,
Packit 90a5c9
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
Packit 90a5c9
 * See the License for the specific language governing permissions and
Packit 90a5c9
 * limitations under the License.
Packit 90a5c9
 */
Packit 90a5c9
Packit 90a5c9
/*
Packit 90a5c9
 * mod_bucketeer.c: split buckets whenever we find a control-char
Packit 90a5c9
 *
Packit 90a5c9
 * Written by Ian Holsman
Packit 90a5c9
 *
Packit 90a5c9
 */
Packit 90a5c9
Packit 90a5c9
#include "httpd.h"
Packit 90a5c9
#include "http_config.h"
Packit 90a5c9
#include "http_log.h"
Packit 90a5c9
#include "apr_strings.h"
Packit 90a5c9
#include "apr_general.h"
Packit 90a5c9
#include "util_filter.h"
Packit 90a5c9
#include "apr_buckets.h"
Packit 90a5c9
#include "http_request.h"
Packit 90a5c9
#include "http_protocol.h"
Packit 90a5c9
Packit 90a5c9
static const char bucketeerFilterName[] = "BUCKETEER";
Packit 90a5c9
module AP_MODULE_DECLARE_DATA bucketeer_module;
Packit 90a5c9
Packit 90a5c9
typedef struct bucketeer_filter_config_t
Packit 90a5c9
{
Packit 90a5c9
    char bucketdelimiter;
Packit 90a5c9
    char passdelimiter;
Packit 90a5c9
    char flushdelimiter;
Packit 90a5c9
} bucketeer_filter_config_t;
Packit 90a5c9
Packit 90a5c9
Packit 90a5c9
static void *create_bucketeer_server_config(apr_pool_t *p, server_rec *s)
Packit 90a5c9
{
Packit 90a5c9
    bucketeer_filter_config_t *c = apr_pcalloc(p, sizeof *c);
Packit 90a5c9
Packit 90a5c9
    c->bucketdelimiter = 0x02; /* ^B */
Packit 90a5c9
    c->passdelimiter = 0x10;   /* ^P */
Packit 90a5c9
    c->flushdelimiter = 0x06;  /* ^F */
Packit 90a5c9
Packit 90a5c9
    return c;
Packit 90a5c9
}
Packit 90a5c9
Packit 90a5c9
typedef struct bucketeer_ctx_t
Packit 90a5c9
{
Packit 90a5c9
    apr_bucket_brigade *bb;
Packit 90a5c9
} bucketeer_ctx_t;
Packit 90a5c9
Packit 90a5c9
static apr_status_t bucketeer_out_filter(ap_filter_t *f,
Packit 90a5c9
                                         apr_bucket_brigade *bb)
Packit 90a5c9
{
Packit 90a5c9
    apr_bucket *e;
Packit 90a5c9
    request_rec *r = f->r;
Packit 90a5c9
    bucketeer_ctx_t *ctx = f->ctx;
Packit 90a5c9
    bucketeer_filter_config_t *c;
Packit 90a5c9
Packit 90a5c9
    c = ap_get_module_config(r->server->module_config, &bucketeer_module);
Packit 90a5c9
Packit 90a5c9
    /* If have a context, it means we've done this before successfully. */
Packit 90a5c9
    if (!ctx) {
Packit 90a5c9
        if (!r->content_type || strncmp(r->content_type, "text/", 5)) {
Packit 90a5c9
            ap_remove_output_filter(f);
Packit 90a5c9
            return ap_pass_brigade(f->next, bb);
Packit 90a5c9
        }
Packit 90a5c9
Packit 90a5c9
        /* We're cool with filtering this. */
Packit 90a5c9
        ctx = f->ctx = apr_pcalloc(f->r->pool, sizeof(*ctx));
Packit 90a5c9
        ctx->bb = apr_brigade_create(f->r->pool, f->c->bucket_alloc);
Packit 90a5c9
        apr_table_unset(f->r->headers_out, "Content-Length");
Packit 90a5c9
    }
Packit 90a5c9
Packit 90a5c9
    for (e = APR_BRIGADE_FIRST(bb);
Packit 90a5c9
         e != APR_BRIGADE_SENTINEL(bb);
Packit 90a5c9
         e = APR_BUCKET_NEXT(e))
Packit 90a5c9
    {
Packit 90a5c9
        const char *data;
Packit 90a5c9
        apr_size_t len, i, lastpos;
Packit 90a5c9
Packit 90a5c9
        if (APR_BUCKET_IS_EOS(e)) {
Packit 90a5c9
            APR_BUCKET_REMOVE(e);
Packit 90a5c9
            APR_BRIGADE_INSERT_TAIL(ctx->bb, e);
Packit 90a5c9
Packit 90a5c9
            /* Okay, we've seen the EOS.
Packit 90a5c9
             * Time to pass it along down the chain.
Packit 90a5c9
             */
Packit 90a5c9
            return ap_pass_brigade(f->next, ctx->bb);
Packit 90a5c9
        }
Packit 90a5c9
Packit 90a5c9
        if (APR_BUCKET_IS_FLUSH(e)) {
Packit 90a5c9
            /*
Packit 90a5c9
             * Ignore flush buckets for the moment..
Packit 90a5c9
             * we decide what to stream
Packit 90a5c9
             */
Packit 90a5c9
            continue;
Packit 90a5c9
        }
Packit 90a5c9
Packit 90a5c9
        if (APR_BUCKET_IS_METADATA(e)) {
Packit 90a5c9
            /* metadata bucket */
Packit 90a5c9
            apr_bucket *cpy;
Packit 90a5c9
            apr_bucket_copy(e, &cpy);
Packit 90a5c9
            APR_BRIGADE_INSERT_TAIL(ctx->bb, cpy);
Packit 90a5c9
            continue;
Packit 90a5c9
        }
Packit 90a5c9
Packit 90a5c9
        /* read */
Packit 90a5c9
        apr_bucket_read(e, &data, &len, APR_BLOCK_READ);
Packit 90a5c9
Packit 90a5c9
        if (len > 0) {
Packit 90a5c9
            lastpos = 0;
Packit 90a5c9
            for (i = 0; i < len; i++) {
Packit 90a5c9
                if (data[i] == c->flushdelimiter ||
Packit 90a5c9
                    data[i] == c->bucketdelimiter ||
Packit 90a5c9
                    data[i] == c->passdelimiter) {
Packit 90a5c9
                    apr_bucket *p;
Packit 90a5c9
                    if (i - lastpos > 0) {
Packit 90a5c9
                        p = apr_bucket_pool_create(apr_pmemdup(f->r->pool,
Packit 90a5c9
                                                               &data[lastpos],
Packit 90a5c9
                                                               i - lastpos),
Packit 90a5c9
                                                    i - lastpos,
Packit 90a5c9
                                                    f->r->pool,
Packit 90a5c9
                                                    f->c->bucket_alloc);
Packit 90a5c9
                        APR_BRIGADE_INSERT_TAIL(ctx->bb, p);
Packit 90a5c9
                    }
Packit 90a5c9
                    lastpos = i + 1;
Packit 90a5c9
                    if (data[i] == c->flushdelimiter) {
Packit 90a5c9
                        p = apr_bucket_flush_create(f->c->bucket_alloc);
Packit 90a5c9
                        APR_BRIGADE_INSERT_TAIL(ctx->bb, p);
Packit 90a5c9
                    }
Packit 90a5c9
                    if (data[i] == c->passdelimiter) {
Packit 90a5c9
                        apr_status_t rv;
Packit 90a5c9
Packit 90a5c9
                        rv = ap_pass_brigade(f->next, ctx->bb);
Packit 90a5c9
                        if (rv) {
Packit 90a5c9
                            return rv;
Packit 90a5c9
                        }
Packit 90a5c9
                    }
Packit 90a5c9
                }
Packit 90a5c9
            }
Packit 90a5c9
            /* XXX: really should append this to the next 'real' bucket */
Packit 90a5c9
            if (lastpos < i) {
Packit 90a5c9
                apr_bucket *p;
Packit 90a5c9
                p = apr_bucket_pool_create(apr_pmemdup(f->r->pool,
Packit 90a5c9
                                                       &data[lastpos],
Packit 90a5c9
                                                       i - lastpos),
Packit 90a5c9
                                           i - lastpos,
Packit 90a5c9
                                           f->r->pool,
Packit 90a5c9
                                           f->c->bucket_alloc);
Packit 90a5c9
                lastpos = i;
Packit 90a5c9
                APR_BRIGADE_INSERT_TAIL(ctx->bb, p);
Packit 90a5c9
            }
Packit 90a5c9
        }
Packit 90a5c9
    }
Packit 90a5c9
Packit 90a5c9
    return APR_SUCCESS;
Packit 90a5c9
}
Packit 90a5c9
Packit 90a5c9
static void register_hooks(apr_pool_t * p)
Packit 90a5c9
{
Packit 90a5c9
    ap_register_output_filter(bucketeerFilterName, bucketeer_out_filter,
Packit 90a5c9
                              NULL, AP_FTYPE_RESOURCE-1);
Packit 90a5c9
}
Packit 90a5c9
Packit 90a5c9
static const command_rec bucketeer_filter_cmds[] = {
Packit 90a5c9
    {NULL}
Packit 90a5c9
};
Packit 90a5c9
Packit 90a5c9
AP_DECLARE_MODULE(bucketeer) = {
Packit 90a5c9
    STANDARD20_MODULE_STUFF,
Packit 90a5c9
    NULL,
Packit 90a5c9
    NULL,
Packit 90a5c9
    create_bucketeer_server_config,
Packit 90a5c9
    NULL,
Packit 90a5c9
    bucketeer_filter_cmds,
Packit 90a5c9
    register_hooks
Packit 90a5c9
};