/* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You under the Apache License, Version 2.0
* (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#include "httpd.h"
#include "http_config.h"
#include "http_request.h"
#include "http_log.h"
module AP_MODULE_DECLARE_DATA allowhandlers_module;
typedef enum {
AH_ALLOW = 0,
AH_DENY = 1,
} ah_op_e;
typedef struct {
apr_table_t *handlers;
ah_op_e op;
} ah_conf_t;
static const char * const forbidden_handler = "forbidden";
static const char * const no_handler = "none";
static int ah_fixups(request_rec *r)
{
ah_conf_t *conf = ap_get_module_config(r->per_dir_config,
&allowhandlers_module);
int match = 0;
const char *handler_name;
if (!r->handler || r->handler[0] == '\0') {
handler_name = no_handler;
}
else if (strcasecmp(r->handler, forbidden_handler) == 0) {
ap_log_rerror(APLOG_MARK, APLOG_DEBUG, 0, r,
APLOGNO(02398) "Handler 'forbidden' denied by "
"server configuration: URI %s (file %s)",
r->uri, r->filename);
return HTTP_FORBIDDEN;
}
else {
handler_name = r->handler;
}
if (!conf)
return DECLINED;
if (conf->handlers && apr_table_get(conf->handlers, handler_name))
match = 1;
if ((match && conf->op == AH_ALLOW) || (!match && conf->op == AH_DENY)) {
return DECLINED;
}
else {
if (handler_name != no_handler) {
ap_log_rerror(APLOG_MARK, APLOG_DEBUG, 0, r,
APLOGNO(02399) "Handler '%s' denied by "
"server configuration: URI %s (file %s)",
r->handler, r->uri, r->filename);
}
else {
ap_log_rerror(APLOG_MARK, APLOG_DEBUG, 0, r,
APLOGNO(02400) "Handler denied by server "
"configuration: No handler set for URI %s (file %s)",
r->uri, r->filename);
}
return HTTP_FORBIDDEN;
}
}
static void *ah_create_conf(apr_pool_t * p, char *dummy)
{
ah_conf_t *conf = apr_pcalloc(p, sizeof(ah_conf_t));
conf->op = AH_DENY;
return conf;
}
static const char *set_allowed_handlers(cmd_parms *cmd, void *d, int argc, char *const argv[])
{
int i;
ah_conf_t* conf = (ah_conf_t*) d;
const char *err = ap_check_cmd_context(cmd, NOT_IN_HTACCESS);
if (err)
return err;
if (argc == 0)
return "AllowHandlers: No handler name given";
conf->op = AH_ALLOW;
if (conf->handlers)
apr_table_clear(conf->handlers);
for (i = 0; i < argc; i++) {
if (strcasecmp(argv[i], forbidden_handler) == 0 && conf->op != AH_DENY)
return "Handler name 'forbidden' cannot be changed.";
if (strcasecmp(argv[i], "all") == 0) {
if (argc != 1)
return "'all' not possible with specific handler names";
conf->op = AH_DENY;
return NULL;
}
else if (strcasecmp(argv[i], "not") == 0) {
if (i != 0 || argc == 1)
return "'not' must come before specific handler names";
conf->op = AH_DENY;
}
else {
if (!conf->handlers)
conf->handlers = apr_table_make(cmd->pool, 4);
apr_table_setn(conf->handlers, argv[i], "1");
}
}
return NULL;
}
static void ah_register_hooks(apr_pool_t * p)
{
ap_hook_fixups(ah_fixups, NULL, NULL, APR_HOOK_REALLY_LAST);
}
static const command_rec ah_cmds[] = {
AP_INIT_TAKE_ARGV("AllowHandlers", set_allowed_handlers, NULL, ACCESS_CONF,
"only allow specific handlers (use 'not' to negate)"),
{NULL}
};
AP_DECLARE_MODULE(allowhandlers) = {
STANDARD20_MODULE_STUFF,
ah_create_conf,
NULL,
NULL,
NULL,
ah_cmds,
ah_register_hooks,
};