Blame server/util_xml.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
#include "apr_xml.h"
Packit 90a5c9
Packit 90a5c9
#include "httpd.h"
Packit 90a5c9
#include "http_protocol.h"
Packit 90a5c9
#include "http_log.h"
Packit 90a5c9
#include "http_core.h"
Packit 90a5c9
Packit 90a5c9
#include "util_charset.h"
Packit 90a5c9
#include "util_xml.h"
Packit 90a5c9
Packit 90a5c9
Packit 90a5c9
/* used for reading input blocks */
Packit 90a5c9
#define READ_BLOCKSIZE 2048
Packit 90a5c9
Packit 90a5c9
Packit 90a5c9
/* we know core's module_index is 0 */
Packit 90a5c9
#undef APLOG_MODULE_INDEX
Packit 90a5c9
#define APLOG_MODULE_INDEX AP_CORE_MODULE_INDEX
Packit 90a5c9
Packit 90a5c9
AP_DECLARE(int) ap_xml_parse_input(request_rec * r, apr_xml_doc **pdoc)
Packit 90a5c9
{
Packit 90a5c9
    apr_xml_parser *parser;
Packit 90a5c9
    apr_bucket_brigade *brigade;
Packit 90a5c9
    int seen_eos;
Packit 90a5c9
    apr_status_t status;
Packit 90a5c9
    char errbuf[200];
Packit 90a5c9
    apr_size_t total_read = 0;
Packit 90a5c9
    apr_size_t limit_xml_body = ap_get_limit_xml_body(r);
Packit 90a5c9
    int result = HTTP_BAD_REQUEST;
Packit 90a5c9
Packit 90a5c9
    parser = apr_xml_parser_create(r->pool);
Packit 90a5c9
    brigade = apr_brigade_create(r->pool, r->connection->bucket_alloc);
Packit 90a5c9
Packit 90a5c9
    seen_eos = 0;
Packit 90a5c9
    total_read = 0;
Packit 90a5c9
Packit 90a5c9
    do {
Packit 90a5c9
        apr_bucket *bucket;
Packit 90a5c9
Packit 90a5c9
        /* read the body, stuffing it into the parser */
Packit 90a5c9
        status = ap_get_brigade(r->input_filters, brigade,
Packit 90a5c9
                                AP_MODE_READBYTES, APR_BLOCK_READ,
Packit 90a5c9
                                READ_BLOCKSIZE);
Packit 90a5c9
Packit 90a5c9
        if (status != APR_SUCCESS) {
Packit 90a5c9
            result = ap_map_http_request_error(status, HTTP_BAD_REQUEST);
Packit 90a5c9
            goto read_error;
Packit 90a5c9
        }
Packit 90a5c9
Packit 90a5c9
        for (bucket = APR_BRIGADE_FIRST(brigade);
Packit 90a5c9
             bucket != APR_BRIGADE_SENTINEL(brigade);
Packit 90a5c9
             bucket = APR_BUCKET_NEXT(bucket))
Packit 90a5c9
        {
Packit 90a5c9
            const char *data;
Packit 90a5c9
            apr_size_t len;
Packit 90a5c9
Packit 90a5c9
            if (APR_BUCKET_IS_EOS(bucket)) {
Packit 90a5c9
                seen_eos = 1;
Packit 90a5c9
                break;
Packit 90a5c9
            }
Packit 90a5c9
Packit 90a5c9
            if (APR_BUCKET_IS_METADATA(bucket)) {
Packit 90a5c9
                continue;
Packit 90a5c9
            }
Packit 90a5c9
Packit 90a5c9
            status = apr_bucket_read(bucket, &data, &len, APR_BLOCK_READ);
Packit 90a5c9
            if (status != APR_SUCCESS) {
Packit 90a5c9
                goto read_error;
Packit 90a5c9
            }
Packit 90a5c9
Packit 90a5c9
            total_read += len;
Packit 90a5c9
            if (limit_xml_body && total_read > limit_xml_body) {
Packit 90a5c9
                ap_log_rerror(APLOG_MARK, APLOG_ERR, 0, r, APLOGNO(00539)
Packit 90a5c9
                              "XML request body is larger than the configured "
Packit 90a5c9
                              "limit of %lu", (unsigned long)limit_xml_body);
Packit 90a5c9
                result = HTTP_REQUEST_ENTITY_TOO_LARGE;
Packit 90a5c9
                goto read_error;
Packit 90a5c9
            }
Packit 90a5c9
Packit 90a5c9
            status = apr_xml_parser_feed(parser, data, len);
Packit 90a5c9
            if (status) {
Packit 90a5c9
                goto parser_error;
Packit 90a5c9
            }
Packit 90a5c9
        }
Packit 90a5c9
Packit 90a5c9
        apr_brigade_cleanup(brigade);
Packit 90a5c9
    } while (!seen_eos);
Packit 90a5c9
Packit 90a5c9
    apr_brigade_destroy(brigade);
Packit 90a5c9
Packit 90a5c9
    /* tell the parser that we're done */
Packit 90a5c9
    status = apr_xml_parser_done(parser, pdoc);
Packit 90a5c9
    if (status) {
Packit 90a5c9
        /* Some parsers are stupid and return an error on blank documents. */
Packit 90a5c9
        if (!total_read) {
Packit 90a5c9
            *pdoc = NULL;
Packit 90a5c9
            return OK;
Packit 90a5c9
        }
Packit 90a5c9
        ap_log_rerror(APLOG_MARK, APLOG_ERR, 0, r, APLOGNO(00540)
Packit 90a5c9
                      "XML parser error (at end). status=%d", status);
Packit 90a5c9
        return HTTP_BAD_REQUEST;
Packit 90a5c9
    }
Packit 90a5c9
Packit 90a5c9
#if APR_CHARSET_EBCDIC
Packit 90a5c9
    apr_xml_parser_convert_doc(r->pool, *pdoc, ap_hdrs_from_ascii);
Packit 90a5c9
#endif
Packit 90a5c9
    return OK;
Packit 90a5c9
Packit 90a5c9
  parser_error:
Packit 90a5c9
    (void) apr_xml_parser_geterror(parser, errbuf, sizeof(errbuf));
Packit 90a5c9
    ap_log_rerror(APLOG_MARK, APLOG_ERR, 0, r, APLOGNO(00541)
Packit 90a5c9
                  "XML Parser Error: %s", errbuf);
Packit 90a5c9
Packit 90a5c9
    /* FALLTHRU */
Packit 90a5c9
Packit 90a5c9
  read_error:
Packit 90a5c9
    /* make sure the parser is terminated */
Packit 90a5c9
    (void) apr_xml_parser_done(parser, NULL);
Packit 90a5c9
Packit 90a5c9
    apr_brigade_destroy(brigade);
Packit 90a5c9
Packit 90a5c9
    /* Apache will supply a default error, plus the error log above. */
Packit 90a5c9
    return result;
Packit 90a5c9
}