mod_reqtimeout.c revision b92eb1e3d9db61b8795f08da9f0dff400ee258a8
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf/* Licensed to the Apache Software Foundation (ASF) under one or more
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf * contributor license agreements. See the NOTICE file distributed with
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf * this work for additional information regarding copyright ownership.
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf * The ASF licenses this file to You under the Apache License, Version 2.0
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf * (the "License"); you may not use this file except in compliance with
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf * the License. You may obtain a copy of the License at
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf *
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf * http://www.apache.org/licenses/LICENSE-2.0
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf *
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf * Unless required by applicable law or agreed to in writing, software
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf * distributed under the License is distributed on an "AS IS" BASIS,
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf * See the License for the specific language governing permissions and
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf * limitations under the License.
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf */
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf#include "httpd.h"
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf#include "http_config.h"
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf#include "http_request.h"
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf#include "http_connection.h"
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf#include "http_protocol.h"
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf#include "http_log.h"
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf#include "http_core.h"
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf#include "util_filter.h"
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf#define APR_WANT_STRFUNC
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf#include "apr_strings.h"
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf#include "apr_version.h"
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsfmodule AP_MODULE_DECLARE_DATA reqtimeout_module;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf#define UNSET -1
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf#define MRT_DEFAULT_HEADER_TIMEOUT 20
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf#define MRT_DEFAULT_HEADER_MAX_TIMEOUT 40
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf#define MRT_DEFAULT_HEADER_MIN_RATE 500
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf#define MRT_DEFAULT_BODY_TIMEOUT 20
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf#define MRT_DEFAULT_BODY_MAX_TIMEOUT 0
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf#define MRT_DEFAULT_BODY_MIN_RATE 500
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsftypedef struct
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf{
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf int header_timeout; /* timeout for reading the req hdrs in secs */
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf int header_max_timeout; /* max timeout for req hdrs in secs */
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf int header_min_rate; /* min rate for reading req hdrs in bytes/s */
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf apr_time_t header_rate_factor;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf int body_timeout; /* timeout for reading the req body in secs */
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf int body_max_timeout; /* max timeout for req body in secs */
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf int body_min_rate; /* min rate for reading req body in bytes/s */
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf apr_time_t body_rate_factor;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf} reqtimeout_srv_cfg;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf/* this struct is used both as conn_config and as filter context */
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsftypedef struct
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf{
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf apr_time_t timeout_at;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf apr_time_t max_timeout_at;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf int min_rate;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf int new_timeout;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf int new_max_timeout;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf int in_keep_alive;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf char *type;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf apr_socket_t *socket;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf apr_time_t rate_factor;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf apr_bucket_brigade *tmpbb;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf} reqtimeout_con_cfg;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsfstatic const char *const reqtimeout_filter_name = "reqtimeout";
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsfstatic int default_header_rate_factor;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsfstatic int default_body_rate_factor;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsfstatic void extend_timeout(reqtimeout_con_cfg *ccfg, apr_bucket_brigade *bb)
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf{
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf apr_off_t len;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf apr_time_t new_timeout_at;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf if (apr_brigade_length(bb, 0, &len) != APR_SUCCESS || len <= 0)
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf return;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf new_timeout_at = ccfg->timeout_at + len * ccfg->rate_factor;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf if (ccfg->max_timeout_at > 0 && new_timeout_at > ccfg->max_timeout_at) {
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf ccfg->timeout_at = ccfg->max_timeout_at;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf }
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf else {
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf ccfg->timeout_at = new_timeout_at;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf }
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf}
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsfstatic apr_status_t check_time_left(reqtimeout_con_cfg *ccfg,
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf apr_time_t *time_left_p,
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf apr_time_t now)
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf{
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf if (!now)
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf now = apr_time_now();
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf *time_left_p = ccfg->timeout_at - now;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf if (*time_left_p <= 0)
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf return APR_TIMEUP;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf if (*time_left_p < apr_time_from_sec(1)) {
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf *time_left_p = apr_time_from_sec(1);
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf }
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf return APR_SUCCESS;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf}
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsfstatic apr_status_t have_lf_or_eos(apr_bucket_brigade *bb)
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf{
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf apr_bucket *b = APR_BRIGADE_LAST(bb);
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf for ( ; b != APR_BRIGADE_SENTINEL(bb) ; b = APR_BUCKET_PREV(b) ) {
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf const char *str;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf apr_size_t len;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf apr_status_t rv;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf if (APR_BUCKET_IS_EOS(b))
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf return APR_SUCCESS;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf if (APR_BUCKET_IS_METADATA(b))
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf continue;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf rv = apr_bucket_read(b, &str, &len, APR_BLOCK_READ);
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf if (rv != APR_SUCCESS)
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf return rv;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf if (len == 0)
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf continue;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf if (str[len-1] == APR_ASCII_LF)
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf return APR_SUCCESS;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf }
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf return APR_INCOMPLETE;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf}
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf/*
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf * Append bbIn to bbOut and merge small buckets, to avoid DoS by high memory
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf * usage
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf */
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsfstatic apr_status_t brigade_append(apr_bucket_brigade *bbOut, apr_bucket_brigade *bbIn)
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf{
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf while (!APR_BRIGADE_EMPTY(bbIn)) {
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf apr_bucket *e = APR_BRIGADE_FIRST(bbIn);
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf const char *str;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf apr_size_t len;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf apr_status_t rv;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf rv = apr_bucket_read(e, &str, &len, APR_BLOCK_READ);
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf if (rv != APR_SUCCESS) {
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf return rv;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf }
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf APR_BUCKET_REMOVE(e);
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf if (APR_BUCKET_IS_METADATA(e) || len > APR_BUCKET_BUFF_SIZE/4) {
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf APR_BRIGADE_INSERT_TAIL(bbOut, e);
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf }
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf else {
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf if (len > 0) {
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf rv = apr_brigade_write(bbOut, NULL, NULL, str, len);
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf if (rv != APR_SUCCESS) {
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf apr_bucket_destroy(e);
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf return rv;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf }
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf }
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf apr_bucket_destroy(e);
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf }
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf }
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf return APR_SUCCESS;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf}
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf#define MIN(x,y) ((x) < (y) ? (x) : (y))
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsfstatic apr_status_t reqtimeout_filter(ap_filter_t *f,
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf apr_bucket_brigade *bb,
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf ap_input_mode_t mode,
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf apr_read_type_e block,
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf apr_off_t readbytes)
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf{
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf apr_time_t time_left;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf apr_time_t now = 0;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf apr_status_t rv;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf apr_interval_time_t saved_sock_timeout = UNSET;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf reqtimeout_con_cfg *ccfg = f->ctx;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf if (ccfg->in_keep_alive) {
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf /* For this read, the normal keep-alive timeout must be used */
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf ccfg->in_keep_alive = 0;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf return ap_get_brigade(f->next, bb, mode, block, readbytes);
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf }
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf if (ccfg->new_timeout > 0) {
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf /* set new timeout */
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf now = apr_time_now();
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf ccfg->timeout_at = now + apr_time_from_sec(ccfg->new_timeout);
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf ccfg->new_timeout = 0;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf if (ccfg->new_max_timeout > 0) {
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf ccfg->max_timeout_at = now + apr_time_from_sec(ccfg->new_max_timeout);
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf ccfg->new_max_timeout = 0;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf }
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf }
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf else if (ccfg->timeout_at == 0) {
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf /* no timeout set */
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf return ap_get_brigade(f->next, bb, mode, block, readbytes);
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf }
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf if (!ccfg->socket) {
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf ccfg->socket = ap_get_conn_socket(f->c);
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf }
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf rv = check_time_left(ccfg, &time_left, now);
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf if (rv != APR_SUCCESS)
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf goto out;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf if (block == APR_NONBLOCK_READ || mode == AP_MODE_INIT
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf || mode == AP_MODE_EATCRLF) {
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf rv = ap_get_brigade(f->next, bb, mode, block, readbytes);
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf if (ccfg->min_rate > 0 && rv == APR_SUCCESS) {
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf extend_timeout(ccfg, bb);
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf }
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf return rv;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf }
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf rv = apr_socket_timeout_get(ccfg->socket, &saved_sock_timeout);
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf AP_DEBUG_ASSERT(rv == APR_SUCCESS);
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf rv = apr_socket_timeout_set(ccfg->socket, MIN(time_left, saved_sock_timeout));
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf AP_DEBUG_ASSERT(rv == APR_SUCCESS);
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf if (mode == AP_MODE_GETLINE) {
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf /*
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf * For a blocking AP_MODE_GETLINE read, apr_brigade_split_line()
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf * would loop until a whole line has been read. As this would make it
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf * impossible to enforce a total timeout, we only do non-blocking
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf * reads.
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf */
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf apr_off_t remaining = HUGE_STRING_LEN;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf do {
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf apr_off_t bblen;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf#if APR_MAJOR_VERSION < 2
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf apr_int32_t nsds;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf apr_interval_time_t poll_timeout;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf apr_pollfd_t pollset;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf#endif
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf rv = ap_get_brigade(f->next, bb, AP_MODE_GETLINE, APR_NONBLOCK_READ, remaining);
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf if (rv != APR_SUCCESS && !APR_STATUS_IS_EAGAIN(rv)) {
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf break;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf }
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf if (!APR_BRIGADE_EMPTY(bb)) {
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf if (ccfg->min_rate > 0) {
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf extend_timeout(ccfg, bb);
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf }
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf rv = have_lf_or_eos(bb);
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf if (rv != APR_INCOMPLETE) {
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf break;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf }
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf rv = apr_brigade_length(bb, 1, &bblen);
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf if (rv != APR_SUCCESS) {
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf break;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf }
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf remaining -= bblen;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf if (remaining <= 0) {
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf break;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf }
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf /* Haven't got a whole line yet, save what we have ... */
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf if (!ccfg->tmpbb) {
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf ccfg->tmpbb = apr_brigade_create(f->c->pool, f->c->bucket_alloc);
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf }
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf rv = brigade_append(ccfg->tmpbb, bb);
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf if (rv != APR_SUCCESS)
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf break;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf }
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf /* ... and wait for more */
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf#if APR_MAJOR_VERSION < 2
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf pollset.p = f->c->pool;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf pollset.desc_type = APR_POLL_SOCKET;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf pollset.reqevents = APR_POLLIN|APR_POLLHUP;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf pollset.desc.s = ccfg->socket;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf apr_socket_timeout_get(ccfg->socket, &poll_timeout);
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf rv = apr_poll(&pollset, 1, &nsds, poll_timeout);
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf#else
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf rv = apr_socket_wait(ccfg->socket, APR_WAIT_READ);
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf#endif
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf if (rv != APR_SUCCESS)
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf break;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf rv = check_time_left(ccfg, &time_left, 0);
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf if (rv != APR_SUCCESS)
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf break;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf rv = apr_socket_timeout_set(ccfg->socket,
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf MIN(time_left, saved_sock_timeout));
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf AP_DEBUG_ASSERT(rv == APR_SUCCESS);
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf } while (1);
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf if (ccfg->tmpbb)
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf APR_BRIGADE_PREPEND(bb, ccfg->tmpbb);
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf }
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf else {
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf /* mode != AP_MODE_GETLINE */
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf rv = ap_get_brigade(f->next, bb, mode, block, readbytes);
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf if (ccfg->min_rate > 0 && rv == APR_SUCCESS) {
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf extend_timeout(ccfg, bb);
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf }
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf }
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf apr_socket_timeout_set(ccfg->socket, saved_sock_timeout);
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsfout:
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf if (APR_STATUS_IS_TIMEUP(rv)) {
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf ap_log_cerror(APLOG_MARK, APLOG_INFO, 0, f->c, APLOGNO(01382)
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf "Request %s read timeout", ccfg->type);
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf /*
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf * If we allow a normal lingering close, the client may keep this
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf * process/thread busy for another 30s (MAX_SECS_TO_LINGER).
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf * Therefore we tell ap_lingering_close() to shorten this period to
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf * 2s (SECONDS_TO_LINGER).
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf */
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf apr_table_setn(f->c->notes, "short-lingering-close", "1");
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf /*
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf * Also, we must not allow keep-alive requests, as
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf * ap_finalize_protocol() may ignore our error status (if the timeout
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf * happened on a request body that is discarded).
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf */
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf f->c->keepalive = AP_CONN_CLOSE;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf }
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf return rv;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf}
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsfstatic int reqtimeout_init(conn_rec *c)
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf{
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf reqtimeout_con_cfg *ccfg;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf reqtimeout_srv_cfg *cfg;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf cfg = ap_get_module_config(c->base_server->module_config,
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf &reqtimeout_module);
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf AP_DEBUG_ASSERT(cfg != NULL);
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf if (cfg->header_timeout == 0 && cfg->body_timeout == 0) {
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf /* disabled for this vhost */
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf return DECLINED;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf }
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf if (cfg->header_timeout == UNSET && cfg->body_timeout == UNSET) {
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf /* if everything is unset, skip by default. */
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf return DECLINED;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf }
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf ccfg = apr_pcalloc(c->pool, sizeof(reqtimeout_con_cfg));
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf ccfg->type = "header";
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf if (cfg->header_timeout != UNSET) {
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf ccfg->new_timeout = cfg->header_timeout;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf ccfg->new_max_timeout = cfg->header_max_timeout;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf ccfg->min_rate = cfg->header_min_rate;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf ccfg->rate_factor = cfg->header_rate_factor;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf }
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf else {
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf ccfg->new_timeout = MRT_DEFAULT_HEADER_TIMEOUT;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf ccfg->new_max_timeout = MRT_DEFAULT_HEADER_MAX_TIMEOUT;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf ccfg->min_rate = MRT_DEFAULT_HEADER_MIN_RATE;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf ccfg->rate_factor = default_header_rate_factor;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf }
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf ap_set_module_config(c->conn_config, &reqtimeout_module, ccfg);
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf ap_add_input_filter("reqtimeout", ccfg, NULL, c);
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf /* we are not handling the connection, we just do initialization */
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf return DECLINED;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf}
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsfstatic int reqtimeout_after_headers(request_rec *r)
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf{
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf reqtimeout_srv_cfg *cfg;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf reqtimeout_con_cfg *ccfg =
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf ap_get_module_config(r->connection->conn_config, &reqtimeout_module);
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf if (ccfg == NULL || r->method_number == M_CONNECT) {
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf /* either disabled for this connection or a CONNECT request */
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf return OK;
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf }
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf cfg = ap_get_module_config(r->connection->base_server->module_config,
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf &reqtimeout_module);
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf AP_DEBUG_ASSERT(cfg != NULL);
4dee28b6fc8fff5efde4e7821aeb6defed3fb84dsf
ccfg->timeout_at = 0;
ccfg->max_timeout_at = 0;
ccfg->type = "body";
if (cfg->body_timeout != UNSET) {
ccfg->new_timeout = cfg->body_timeout;
ccfg->new_max_timeout = cfg->body_max_timeout;
ccfg->min_rate = cfg->body_min_rate;
ccfg->rate_factor = cfg->body_rate_factor;
}
else {
ccfg->new_timeout = MRT_DEFAULT_BODY_TIMEOUT;
ccfg->new_max_timeout = MRT_DEFAULT_BODY_MAX_TIMEOUT;
ccfg->min_rate = MRT_DEFAULT_BODY_MIN_RATE;
ccfg->rate_factor = default_body_rate_factor;
}
return OK;
}
static int reqtimeout_after_body(request_rec *r)
{
reqtimeout_srv_cfg *cfg;
reqtimeout_con_cfg *ccfg =
ap_get_module_config(r->connection->conn_config, &reqtimeout_module);
if (ccfg == NULL) {
/* not configured for this connection */
return OK;
}
cfg = ap_get_module_config(r->connection->base_server->module_config,
&reqtimeout_module);
AP_DEBUG_ASSERT(cfg != NULL);
ccfg->timeout_at = 0;
ccfg->max_timeout_at = 0;
ccfg->in_keep_alive = 1;
ccfg->type = "header";
if (ccfg->new_timeout != UNSET) {
ccfg->new_timeout = cfg->header_timeout;
ccfg->new_max_timeout = cfg->header_max_timeout;
ccfg->min_rate = cfg->header_min_rate;
ccfg->rate_factor = cfg->header_rate_factor;
}
else {
ccfg->new_timeout = MRT_DEFAULT_HEADER_TIMEOUT;
ccfg->new_max_timeout = MRT_DEFAULT_HEADER_MAX_TIMEOUT;
ccfg->min_rate = MRT_DEFAULT_HEADER_MIN_RATE;
ccfg->rate_factor = default_header_rate_factor;
}
return OK;
}
static void *reqtimeout_create_srv_config(apr_pool_t *p, server_rec *s)
{
reqtimeout_srv_cfg *cfg = apr_pcalloc(p, sizeof(reqtimeout_srv_cfg));
cfg->header_timeout = UNSET;
cfg->header_max_timeout = UNSET;
cfg->header_min_rate = UNSET;
cfg->body_timeout = UNSET;
cfg->body_max_timeout = UNSET;
cfg->body_min_rate = UNSET;
return cfg;
}
#define MERGE_INT(cfg, b, a, val) cfg->val = (a->val == UNSET) ? b->val : a->val;
static void *reqtimeout_merge_srv_config(apr_pool_t *p, void *base_, void *add_)
{
reqtimeout_srv_cfg *base = base_;
reqtimeout_srv_cfg *add = add_;
reqtimeout_srv_cfg *cfg = apr_pcalloc(p, sizeof(reqtimeout_srv_cfg));
MERGE_INT(cfg, base, add, header_timeout);
MERGE_INT(cfg, base, add, header_max_timeout);
MERGE_INT(cfg, base, add, header_min_rate);
MERGE_INT(cfg, base, add, body_timeout);
MERGE_INT(cfg, base, add, body_max_timeout);
MERGE_INT(cfg, base, add, body_min_rate);
cfg->header_rate_factor = (cfg->header_min_rate == UNSET) ?
base->header_rate_factor : add->header_rate_factor;
cfg->body_rate_factor = (cfg->body_min_rate == UNSET) ?
base->body_rate_factor : add->body_rate_factor;
return cfg;
}
static const char *parse_int(apr_pool_t *p, const char *arg, int *val) {
char *endptr;
*val = strtol(arg, &endptr, 10);
if (arg == endptr) {
return apr_psprintf(p, "Value '%s' not numerical", endptr);
}
if (*endptr != '\0') {
return apr_psprintf(p, "Cannot parse '%s'", endptr);
}
if (*val < 0) {
return "Value must be non-negative";
}
return NULL;
}
static const char *set_reqtimeout_param(reqtimeout_srv_cfg *conf,
apr_pool_t *p,
const char *key,
const char *val)
{
const char *ret = NULL;
char *rate_str = NULL, *initial_str, *max_str = NULL;
int rate = 0, initial = 0, max = 0;
enum { PARAM_HEADER, PARAM_BODY } type;
if (!strcasecmp(key, "header")) {
type = PARAM_HEADER;
}
else if (!strcasecmp(key, "body")) {
type = PARAM_BODY;
}
else {
return "Unknown RequestReadTimeout parameter";
}
if ((rate_str = ap_strcasestr(val, ",minrate="))) {
initial_str = apr_pstrndup(p, val, rate_str - val);
rate_str += strlen(",minrate=");
ret = parse_int(p, rate_str, &rate);
if (ret)
return ret;
if (rate == 0)
return "Minimum data rate must be larger than 0";
if ((max_str = strchr(initial_str, '-'))) {
*max_str++ = '\0';
ret = parse_int(p, max_str, &max);
if (ret)
return ret;
}
ret = parse_int(p, initial_str, &initial);
}
else {
if (ap_strchr_c(val, '-'))
return "Must set MinRate option if using timeout range";
ret = parse_int(p, val, &initial);
}
if (ret)
return ret;
if (max && initial >= max) {
return "Maximum timeout must be larger than initial timeout";
}
if (type == PARAM_HEADER) {
conf->header_timeout = initial;
conf->header_max_timeout = max;
conf->header_min_rate = rate;
if (rate)
conf->header_rate_factor = apr_time_from_sec(1) / rate;
}
else {
conf->body_timeout = initial;
conf->body_max_timeout = max;
conf->body_min_rate = rate;
if (rate)
conf->body_rate_factor = apr_time_from_sec(1) / rate;
}
return ret;
}
static const char *set_reqtimeouts(cmd_parms *cmd, void *mconfig,
const char *arg)
{
reqtimeout_srv_cfg *conf =
ap_get_module_config(cmd->server->module_config,
&reqtimeout_module);
while (*arg) {
char *word, *val;
const char *err;
word = ap_getword_conf(cmd->temp_pool, &arg);
val = strchr(word, '=');
if (!val) {
return "Invalid RequestReadTimeout parameter. Parameter must be "
"in the form 'key=value'";
}
else
*val++ = '\0';
err = set_reqtimeout_param(conf, cmd->pool, word, val);
if (err)
return apr_psprintf(cmd->temp_pool, "RequestReadTimeout: %s=%s: %s",
word, val, err);
}
return NULL;
}
static void reqtimeout_hooks(apr_pool_t *pool)
{
/*
* mod_ssl is AP_FTYPE_CONNECTION + 5 and mod_reqtimeout needs to
* be called before mod_ssl. Otherwise repeated reads during the ssl
* handshake can prevent the timeout from triggering.
*/
ap_register_input_filter(reqtimeout_filter_name, reqtimeout_filter, NULL,
AP_FTYPE_CONNECTION + 8);
/*
* mod_reqtimeout needs to be called before ap_process_http_request (which
* is run at APR_HOOK_REALLY_LAST) but after all other protocol modules.
* This ensures that it only influences normal http connections and not
* e.g. mod_ftp. Also, if mod_reqtimeout used the pre_connection hook, it
* would be inserted on mod_proxy's backend connections.
*/
ap_hook_process_connection(reqtimeout_init, NULL, NULL, APR_HOOK_LAST);
ap_hook_post_read_request(reqtimeout_after_headers, NULL, NULL,
APR_HOOK_MIDDLE);
ap_hook_log_transaction(reqtimeout_after_body, NULL, NULL,
APR_HOOK_MIDDLE);
#if MRT_DEFAULT_HEADER_MIN_RATE > 0
default_header_rate_factor = apr_time_from_sec(1) / MRT_DEFAULT_HEADER_MIN_RATE;
#endif
#if MRT_DEFAULT_BODY_MIN_RATE > 0
default_body_rate_factor = apr_time_from_sec(1) / MRT_DEFAULT_BODY_MIN_RATE;
#endif
}
static const command_rec reqtimeout_cmds[] = {
AP_INIT_RAW_ARGS("RequestReadTimeout", set_reqtimeouts, NULL, RSRC_CONF,
"Set various timeout parameters for reading request "
"headers and body"),
{NULL}
};
AP_DECLARE_MODULE(reqtimeout) = {
STANDARD20_MODULE_STUFF,
NULL, /* create per-dir config structures */
NULL, /* merge per-dir config structures */
reqtimeout_create_srv_config, /* create per-server config structures */
reqtimeout_merge_srv_config, /* merge per-server config structures */
reqtimeout_cmds, /* table of config file commands */
reqtimeout_hooks
};