1 /* Licensed to the Apache Software Foundation (ASF) under one or more
2 * contributor license agreements. See the NOTICE file distributed with
3 * this work for additional information regarding copyright ownership.
4 * The ASF licenses this file to You under the Apache License, Version 2.0
5 * (the "License"); you may not use this file except in compliance with
6 * the License. You may obtain a copy of the License at
8 * http://www.apache.org/licenses/LICENSE-2.0
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
20 #include "http_protocol.h"
22 #include "http_core.h"
24 #include "util_charset.h"
28 /* used for reading input blocks */
29 #define READ_BLOCKSIZE 2048
32 /* we know core's module_index is 0 */
33 #undef APLOG_MODULE_INDEX
34 #define APLOG_MODULE_INDEX AP_CORE_MODULE_INDEX
36 AP_DECLARE(int) ap_xml_parse_input(request_rec * r, apr_xml_doc **pdoc)
38 apr_xml_parser *parser;
39 apr_bucket_brigade *brigade;
43 apr_size_t total_read = 0;
44 apr_size_t limit_xml_body = ap_get_limit_xml_body(r);
45 int result = HTTP_BAD_REQUEST;
47 parser = apr_xml_parser_create(r->pool);
48 brigade = apr_brigade_create(r->pool, r->connection->bucket_alloc);
56 /* read the body, stuffing it into the parser */
57 status = ap_get_brigade(r->input_filters, brigade,
58 AP_MODE_READBYTES, APR_BLOCK_READ,
61 if (status != APR_SUCCESS) {
62 result = ap_map_http_request_error(status, HTTP_BAD_REQUEST);
66 for (bucket = APR_BRIGADE_FIRST(brigade);
67 bucket != APR_BRIGADE_SENTINEL(brigade);
68 bucket = APR_BUCKET_NEXT(bucket))
73 if (APR_BUCKET_IS_EOS(bucket)) {
78 if (APR_BUCKET_IS_METADATA(bucket)) {
82 status = apr_bucket_read(bucket, &data, &len, APR_BLOCK_READ);
83 if (status != APR_SUCCESS) {
88 if (limit_xml_body && total_read > limit_xml_body) {
89 ap_log_rerror(APLOG_MARK, APLOG_ERR, 0, r, APLOGNO(00539)
90 "XML request body is larger than the configured "
91 "limit of %lu", (unsigned long)limit_xml_body);
92 result = HTTP_REQUEST_ENTITY_TOO_LARGE;
96 status = apr_xml_parser_feed(parser, data, len);
102 apr_brigade_cleanup(brigade);
105 apr_brigade_destroy(brigade);
107 /* tell the parser that we're done */
108 status = apr_xml_parser_done(parser, pdoc);
110 /* Some parsers are stupid and return an error on blank documents. */
115 ap_log_rerror(APLOG_MARK, APLOG_ERR, 0, r, APLOGNO(00540)
116 "XML parser error (at end). status=%d", status);
117 return HTTP_BAD_REQUEST;
120 #if APR_CHARSET_EBCDIC
121 apr_xml_parser_convert_doc(r->pool, *pdoc, ap_hdrs_from_ascii);
126 (void) apr_xml_parser_geterror(parser, errbuf, sizeof(errbuf));
127 ap_log_rerror(APLOG_MARK, APLOG_ERR, 0, r, APLOGNO(00541)
128 "XML Parser Error: %s", errbuf);
133 /* make sure the parser is terminated */
134 (void) apr_xml_parser_done(parser, NULL);
136 apr_brigade_destroy(brigade);
138 /* Apache will supply a default error, plus the error log above. */