236 lines
9.8 KiB
Diff
236 lines
9.8 KiB
Diff
diff --git a/docs/manual/mod/core.html.en b/docs/manual/mod/core.html.en
|
|
index 0a24bc8..20d1e5a 100644
|
|
--- a/docs/manual/mod/core.html.en
|
|
+++ b/docs/manual/mod/core.html.en
|
|
@@ -97,6 +97,7 @@ available</td></tr>
|
|
<li><img alt="" src="../images/down.gif" /> <a href="#maxrangeoverlaps">MaxRangeOverlaps</a></li>
|
|
<li><img alt="" src="../images/down.gif" /> <a href="#maxrangereversals">MaxRangeReversals</a></li>
|
|
<li><img alt="" src="../images/down.gif" /> <a href="#maxranges">MaxRanges</a></li>
|
|
+<li><img alt="" src="../images/down.gif" /> <a href="#mergeslashes">MergeSlashes</a></li>
|
|
<li><img alt="" src="../images/down.gif" /> <a href="#mergetrailers">MergeTrailers</a></li>
|
|
<li><img alt="" src="../images/down.gif" /> <a href="#mutex">Mutex</a></li>
|
|
<li><img alt="" src="../images/down.gif" /> <a href="#namevirtualhost">NameVirtualHost</a></li>
|
|
@@ -3465,6 +3466,30 @@ resource </td></tr>
|
|
|
|
</div>
|
|
<div class="top"><a href="#page-header"><img alt="top" src="../images/up.gif" /></a></div>
|
|
+<div class="directive-section"><h2><a name="MergeSlashes" id="MergeSlashes">MergeSlashes</a> <a name="mergeslashes" id="mergeslashes">Directive</a></h2>
|
|
+<table class="directive">
|
|
+<tr><th><a href="directive-dict.html#Description">Description:</a></th><td>Controls whether the server merges consecutive slashes in URLs. </td></tr>
|
|
+<tr><th><a href="directive-dict.html#Syntax">Syntax:</a></th><td><code>MergeSlashes ON | OFF</code></td></tr>
|
|
+<tr><th><a href="directive-dict.html#Default">Default:</a></th><td><code>MergeSlashes ON</code></td></tr>
|
|
+<tr><th><a href="directive-dict.html#Context">Context:</a></th><td>server config, virtual host</td></tr>
|
|
+<tr><th><a href="directive-dict.html#Status">Status:</a></th><td>Core</td></tr>
|
|
+<tr><th><a href="directive-dict.html#Module">Module:</a></th><td>core</td></tr>
|
|
+<tr><th><a href="directive-dict.html#Compatibility">Compatibility:</a></th><td>Available in Apache HTTP Server 2.4.6 in Red Hat Enterprise Linux 7</td></tr>
|
|
+</table>
|
|
+ <p>By default, the server merges (or collapses) multiple consecutive slash
|
|
+ ('/') characters in the path component of the request URL.</p>
|
|
+
|
|
+ <p>When mapping URL's to the filesystem, these multiple slashes are not
|
|
+ significant. However, URL's handled other ways, such as by CGI or proxy,
|
|
+ might prefer to retain the significance of multiple consecutive slashes.
|
|
+ In these cases <code class="directive">MergeSlashes</code> can be set to
|
|
+ <em>OFF</em> to retain the multiple consecutive slashes. In these
|
|
+ configurations, regular expressions used in the configuration file that match
|
|
+ the path component of the URL (<code class="directive">LocationMatch</code>,
|
|
+ <code class="directive">RewriteRule</code>, ...) need to take into account multiple
|
|
+ consecutive slashes.</p>
|
|
+</div>
|
|
+<div class="top"><a href="#page-header"><img alt="top" src="../images/up.gif" /></a></div>
|
|
<div class="directive-section"><h2><a name="MergeTrailers" id="MergeTrailers">MergeTrailers</a> <a name="mergetrailers" id="mergetrailers">Directive</a></h2>
|
|
<table class="directive">
|
|
<tr><th><a href="directive-dict.html#Description">Description:</a></th><td>Determines whether trailers are merged into headers</td></tr>
|
|
--- a/include/http_core.h 2019/03/18 08:49:19 1855736
|
|
+++ b/include/http_core.h 2019/03/18 08:49:59 1855737
|
|
@@ -740,7 +740,7 @@
|
|
#define AP_HTTP_METHODS_LENIENT 1
|
|
#define AP_HTTP_METHODS_REGISTERED 2
|
|
char http_methods;
|
|
-
|
|
+ unsigned int merge_slashes;
|
|
} core_server_config;
|
|
|
|
/* for AddOutputFiltersByType in core.c */
|
|
diff --git a/include/httpd.h b/include/httpd.h
|
|
index 65392f8..99f7f04 100644
|
|
--- a/include/httpd.h
|
|
+++ b/include/httpd.h
|
|
@@ -1697,11 +1697,21 @@ AP_DECLARE(int) ap_unescape_url_keep2f(char *url, int decode_slashes);
|
|
AP_DECLARE(int) ap_unescape_urlencoded(char *query);
|
|
|
|
/**
|
|
- * Convert all double slashes to single slashes
|
|
- * @param name The string to convert
|
|
+ * Convert all double slashes to single slashes, except where significant
|
|
+ * to the filesystem on the current platform.
|
|
+ * @param name The string to convert, assumed to be a filesystem path
|
|
*/
|
|
AP_DECLARE(void) ap_no2slash(char *name);
|
|
|
|
+/**
|
|
+ * Convert all double slashes to single slashes, except where significant
|
|
+ * to the filesystem on the current platform.
|
|
+ * @param name The string to convert
|
|
+ * @param is_fs_path if set to 0, the significance of any double-slashes is
|
|
+ * ignored.
|
|
+ */
|
|
+AP_DECLARE(void) ap_no2slash_ex(char *name, int is_fs_path);
|
|
+
|
|
/**
|
|
* Remove all ./ and xx/../ substrings from a file name. Also remove
|
|
* any leading ../ or /../ substrings.
|
|
diff --git a/server/request.c b/server/request.c
|
|
index dbe3e07..d5c558a 100644
|
|
--- a/server/request.c
|
|
+++ b/server/request.c
|
|
@@ -167,6 +167,8 @@ AP_DECLARE(int) ap_process_request_internal(request_rec *r)
|
|
int file_req = (r->main && r->filename);
|
|
int access_status;
|
|
core_dir_config *d;
|
|
+ core_server_config *sconf =
|
|
+ ap_get_core_module_config(r->server->module_config);
|
|
|
|
/* Ignore embedded %2F's in path for proxy requests */
|
|
if (!r->proxyreq && r->parsed_uri.path) {
|
|
@@ -191,6 +193,12 @@ AP_DECLARE(int) ap_process_request_internal(request_rec *r)
|
|
}
|
|
|
|
ap_getparents(r->uri); /* OK --- shrinking transformations... */
|
|
+ if (sconf->merge_slashes != AP_CORE_CONFIG_OFF) {
|
|
+ ap_no2slash(r->uri);
|
|
+ if (r->parsed_uri.path) {
|
|
+ ap_no2slash(r->parsed_uri.path);
|
|
+ }
|
|
+ }
|
|
|
|
/* All file subrequests are a huge pain... they cannot bubble through the
|
|
* next several steps. Only file subrequests are allowed an empty uri,
|
|
@@ -1411,20 +1419,7 @@ AP_DECLARE(int) ap_location_walk(request_rec *r)
|
|
|
|
cache = prep_walk_cache(AP_NOTE_LOCATION_WALK, r);
|
|
cached = (cache->cached != NULL);
|
|
-
|
|
- /* Location and LocationMatch differ on their behaviour w.r.t. multiple
|
|
- * slashes. Location matches multiple slashes with a single slash,
|
|
- * LocationMatch doesn't. An exception, for backwards brokenness is
|
|
- * absoluteURIs... in which case neither match multiple slashes.
|
|
- */
|
|
- if (r->uri[0] != '/') {
|
|
- entry_uri = r->uri;
|
|
- }
|
|
- else {
|
|
- char *uri = apr_pstrdup(r->pool, r->uri);
|
|
- ap_no2slash(uri);
|
|
- entry_uri = uri;
|
|
- }
|
|
+ entry_uri = r->uri;
|
|
|
|
/* If we have an cache->cached location that matches r->uri,
|
|
* and the vhost's list of locations hasn't changed, we can skip
|
|
@@ -1491,7 +1486,7 @@ AP_DECLARE(int) ap_location_walk(request_rec *r)
|
|
pmatch = apr_palloc(rxpool, nmatch*sizeof(ap_regmatch_t));
|
|
}
|
|
|
|
- if (ap_regexec(entry_core->r, r->uri, nmatch, pmatch, 0)) {
|
|
+ if (ap_regexec(entry_core->r, entry_uri, nmatch, pmatch, 0)) {
|
|
continue;
|
|
}
|
|
|
|
@@ -1501,7 +1496,7 @@ AP_DECLARE(int) ap_location_walk(request_rec *r)
|
|
apr_table_setn(r->subprocess_env,
|
|
((const char **)entry_core->refs->elts)[i],
|
|
apr_pstrndup(r->pool,
|
|
- r->uri + pmatch[i].rm_so,
|
|
+ entry_uri + pmatch[i].rm_so,
|
|
pmatch[i].rm_eo - pmatch[i].rm_so));
|
|
}
|
|
}
|
|
diff --git a/server/util.c b/server/util.c
|
|
index fd7a0a1..e0c558c 100644
|
|
--- a/server/util.c
|
|
+++ b/server/util.c
|
|
@@ -561,16 +561,20 @@ AP_DECLARE(void) ap_getparents(char *name)
|
|
name[l] = '\0';
|
|
}
|
|
}
|
|
-
|
|
-AP_DECLARE(void) ap_no2slash(char *name)
|
|
+AP_DECLARE(void) ap_no2slash_ex(char *name, int is_fs_path)
|
|
{
|
|
+
|
|
char *d, *s;
|
|
|
|
+ if (!*name) {
|
|
+ return;
|
|
+ }
|
|
+
|
|
s = d = name;
|
|
|
|
#ifdef HAVE_UNC_PATHS
|
|
/* Check for UNC names. Leave leading two slashes. */
|
|
- if (s[0] == '/' && s[1] == '/')
|
|
+ if (is_fs_path && s[0] == '/' && s[1] == '/')
|
|
*d++ = *s++;
|
|
#endif
|
|
|
|
@@ -587,6 +591,10 @@ AP_DECLARE(void) ap_no2slash(char *name)
|
|
*d = '\0';
|
|
}
|
|
|
|
+AP_DECLARE(void) ap_no2slash(char *name)
|
|
+{
|
|
+ ap_no2slash_ex(name, 1);
|
|
+}
|
|
|
|
/*
|
|
* copy at most n leading directories of s into d
|
|
diff --git a/server/core.c b/server/core.c
|
|
index b5ab429..a31f1e4 100644
|
|
--- a/server/core.c
|
|
+++ b/server/core.c
|
|
@@ -493,6 +493,7 @@ static void *create_core_server_config(apr_pool_t *a, server_rec *s)
|
|
*/
|
|
|
|
conf->trace_enable = AP_TRACE_UNSET;
|
|
+ conf->merge_slashes = AP_CORE_CONFIG_UNSET;
|
|
|
|
conf->protocols = apr_array_make(a, 5, sizeof(const char *));
|
|
conf->protocols_honor_order = -1;
|
|
@@ -561,7 +562,9 @@ static void *merge_core_server_configs(apr_pool_t *p, void *basev, void *virtv)
|
|
conf->protocols_honor_order = ((virt->protocols_honor_order < 0)?
|
|
base->protocols_honor_order :
|
|
virt->protocols_honor_order);
|
|
-
|
|
+
|
|
+ AP_CORE_MERGE_FLAG(merge_slashes, conf, base, virt);
|
|
+
|
|
return conf;
|
|
}
|
|
|
|
@@ -1872,6 +1875,13 @@ static const char *set_qualify_redirect_url(cmd_parms *cmd, void *d_, int flag)
|
|
return NULL;
|
|
}
|
|
|
|
+static const char *set_core_server_flag(cmd_parms *cmd, void *s_, int flag)
|
|
+{
|
|
+ core_server_config *conf =
|
|
+ ap_get_core_module_config(cmd->server->module_config);
|
|
+ return ap_set_flag_slot(cmd, conf, flag);
|
|
+}
|
|
+
|
|
static const char *set_override_list(cmd_parms *cmd, void *d_, int argc, char *const argv[])
|
|
{
|
|
core_dir_config *d = d_;
|
|
@@ -4598,6 +4608,10 @@ AP_INIT_ITERATE("HttpProtocolOptions", set_http_protocol_options, NULL, RSRC_CON
|
|
"'Unsafe' or 'Strict' (default). Sets HTTP acceptance rules"),
|
|
AP_INIT_ITERATE("RegisterHttpMethod", set_http_method, NULL, RSRC_CONF,
|
|
"Registers non-standard HTTP methods"),
|
|
+AP_INIT_FLAG("MergeSlashes", set_core_server_flag,
|
|
+ (void *)APR_OFFSETOF(core_server_config, merge_slashes),
|
|
+ RSRC_CONF,
|
|
+ "Controls whether consecutive slashes in the URI path are merged"),
|
|
{ NULL }
|
|
};
|
|
|