提交 f2771efd 编写于 作者: J Jonathan Nieder

Git 2.23.3

This merges up the security fix from v2.17.5.
Signed-off-by: NJonathan Nieder <jrnieder@gmail.com>
Git v2.17.5 Release Notes
=========================
This release is to address a security issue: CVE-2020-11008
Fixes since v2.17.4
-------------------
* With a crafted URL that contains a newline or empty host, or lacks
a scheme, the credential helper machinery can be fooled into
providing credential information that is not appropriate for the
protocol in use and host being contacted.
Unlike the vulnerability CVE-2020-5260 fixed in v2.17.4, the
credentials are not for a host of the attacker's choosing; instead,
they are for some unspecified host (based on how the configured
credential helper handles an absent "host" parameter).
The attack has been made impossible by refusing to work with
under-specified credential patterns.
Credit for finding the vulnerability goes to Carlo Arenas.
Git v2.18.4 Release Notes
=========================
This release merges the security fix that appears in v2.17.5; see
the release notes for that version for details.
Git v2.19.5 Release Notes
=========================
This release merges the security fix that appears in v2.17.5; see
the release notes for that version for details.
Git v2.20.4 Release Notes
=========================
This release merges the security fix that appears in v2.17.5; see
the release notes for that version for details.
Git v2.21.3 Release Notes
=========================
This release merges the security fix that appears in v2.17.5; see
the release notes for that version for details.
Git v2.22.4 Release Notes
=========================
This release merges the security fix that appears in v2.17.5; see
the release notes for that version for details.
Git v2.23.3 Release Notes
=========================
This release merges the security fix that appears in v2.17.5; see
the release notes for that version for details.
#!/bin/sh #!/bin/sh
GVF=GIT-VERSION-FILE GVF=GIT-VERSION-FILE
DEF_VER=v2.23.2 DEF_VER=v2.23.3
LF=' LF='
' '
......
Documentation/RelNotes/2.23.2.txt Documentation/RelNotes/2.23.3.txt
\ No newline at end of file \ No newline at end of file
...@@ -89,6 +89,11 @@ static int proto_is_http(const char *s) ...@@ -89,6 +89,11 @@ static int proto_is_http(const char *s)
static void credential_apply_config(struct credential *c) static void credential_apply_config(struct credential *c)
{ {
if (!c->host)
die(_("refusing to work with credential missing host field"));
if (!c->protocol)
die(_("refusing to work with credential missing protocol field"));
if (c->configured) if (c->configured)
return; return;
git_config(credential_config_callback, c); git_config(credential_config_callback, c);
...@@ -191,8 +196,11 @@ int credential_read(struct credential *c, FILE *fp) ...@@ -191,8 +196,11 @@ int credential_read(struct credential *c, FILE *fp)
return 0; return 0;
} }
static void credential_write_item(FILE *fp, const char *key, const char *value) static void credential_write_item(FILE *fp, const char *key, const char *value,
int required)
{ {
if (!value && required)
BUG("credential value for %s is missing", key);
if (!value) if (!value)
return; return;
if (strchr(value, '\n')) if (strchr(value, '\n'))
...@@ -202,11 +210,11 @@ static void credential_write_item(FILE *fp, const char *key, const char *value) ...@@ -202,11 +210,11 @@ static void credential_write_item(FILE *fp, const char *key, const char *value)
void credential_write(const struct credential *c, FILE *fp) void credential_write(const struct credential *c, FILE *fp)
{ {
credential_write_item(fp, "protocol", c->protocol); credential_write_item(fp, "protocol", c->protocol, 1);
credential_write_item(fp, "host", c->host); credential_write_item(fp, "host", c->host, 1);
credential_write_item(fp, "path", c->path); credential_write_item(fp, "path", c->path, 0);
credential_write_item(fp, "username", c->username); credential_write_item(fp, "username", c->username, 0);
credential_write_item(fp, "password", c->password); credential_write_item(fp, "password", c->password, 0);
} }
static int run_credential_helper(struct credential *c, static int run_credential_helper(struct credential *c,
...@@ -352,8 +360,11 @@ int credential_from_url_gently(struct credential *c, const char *url, ...@@ -352,8 +360,11 @@ int credential_from_url_gently(struct credential *c, const char *url,
* (3) proto://<user>:<pass>@<host>/... * (3) proto://<user>:<pass>@<host>/...
*/ */
proto_end = strstr(url, "://"); proto_end = strstr(url, "://");
if (!proto_end) if (!proto_end || proto_end == url) {
return 0; if (!quiet)
warning(_("url has no scheme: %s"), url);
return -1;
}
cp = proto_end + 3; cp = proto_end + 3;
at = strchr(cp, '@'); at = strchr(cp, '@');
colon = strchr(cp, ':'); colon = strchr(cp, ':');
...@@ -374,10 +385,8 @@ int credential_from_url_gently(struct credential *c, const char *url, ...@@ -374,10 +385,8 @@ int credential_from_url_gently(struct credential *c, const char *url,
host = at + 1; host = at + 1;
} }
if (proto_end - url > 0) c->protocol = xmemdupz(url, proto_end - url);
c->protocol = xmemdupz(url, proto_end - url); c->host = url_decode_mem(host, slash - host);
if (slash - host > 0)
c->host = url_decode_mem(host, slash - host);
/* Trim leading and trailing slashes from path */ /* Trim leading and trailing slashes from path */
while (*slash == '/') while (*slash == '/')
slash++; slash++;
...@@ -401,8 +410,6 @@ int credential_from_url_gently(struct credential *c, const char *url, ...@@ -401,8 +410,6 @@ int credential_from_url_gently(struct credential *c, const char *url,
void credential_from_url(struct credential *c, const char *url) void credential_from_url(struct credential *c, const char *url)
{ {
if (credential_from_url_gently(c, url, 0) < 0) { if (credential_from_url_gently(c, url, 0) < 0)
warning(_("skipping credential lookup for url: %s"), url); die(_("credential url cannot be parsed: %s"), url);
credential_clear(c);
}
} }
...@@ -9,6 +9,7 @@ ...@@ -9,6 +9,7 @@
#include "tag.h" #include "tag.h"
#include "fsck.h" #include "fsck.h"
#include "refs.h" #include "refs.h"
#include "url.h"
#include "utf8.h" #include "utf8.h"
#include "decorate.h" #include "decorate.h"
#include "oidset.h" #include "oidset.h"
...@@ -948,17 +949,147 @@ static int fsck_tag(struct tag *tag, const char *data, ...@@ -948,17 +949,147 @@ static int fsck_tag(struct tag *tag, const char *data,
return fsck_tag_buffer(tag, data, size, options); return fsck_tag_buffer(tag, data, size, options);
} }
/*
* Like builtin/submodule--helper.c's starts_with_dot_slash, but without
* relying on the platform-dependent is_dir_sep helper.
*
* This is for use in checking whether a submodule URL is interpreted as
* relative to the current directory on any platform, since \ is a
* directory separator on Windows but not on other platforms.
*/
static int starts_with_dot_slash(const char *str)
{
return str[0] == '.' && (str[1] == '/' || str[1] == '\\');
}
/*
* Like starts_with_dot_slash, this is a variant of submodule--helper's
* helper of the same name with the twist that it accepts backslash as a
* directory separator even on non-Windows platforms.
*/
static int starts_with_dot_dot_slash(const char *str)
{
return str[0] == '.' && starts_with_dot_slash(str + 1);
}
static int submodule_url_is_relative(const char *url)
{
return starts_with_dot_slash(url) || starts_with_dot_dot_slash(url);
}
/*
* Count directory components that a relative submodule URL should chop
* from the remote_url it is to be resolved against.
*
* In other words, this counts "../" components at the start of a
* submodule URL.
*
* Returns the number of directory components to chop and writes a
* pointer to the next character of url after all leading "./" and
* "../" components to out.
*/
static int count_leading_dotdots(const char *url, const char **out)
{
int result = 0;
while (1) {
if (starts_with_dot_dot_slash(url)) {
result++;
url += strlen("../");
continue;
}
if (starts_with_dot_slash(url)) {
url += strlen("./");
continue;
}
*out = url;
return result;
}
}
/*
* Check whether a transport is implemented by git-remote-curl.
*
* If it is, returns 1 and writes the URL that would be passed to
* git-remote-curl to the "out" parameter.
*
* Otherwise, returns 0 and leaves "out" untouched.
*
* Examples:
* http::https://example.com/repo.git -> 1, https://example.com/repo.git
* https://example.com/repo.git -> 1, https://example.com/repo.git
* git://example.com/repo.git -> 0
*
* This is for use in checking for previously exploitable bugs that
* required a submodule URL to be passed to git-remote-curl.
*/
static int url_to_curl_url(const char *url, const char **out)
{
/*
* We don't need to check for case-aliases, "http.exe", and so
* on because in the default configuration, is_transport_allowed
* prevents URLs with those schemes from being cloned
* automatically.
*/
if (skip_prefix(url, "http::", out) ||
skip_prefix(url, "https::", out) ||
skip_prefix(url, "ftp::", out) ||
skip_prefix(url, "ftps::", out))
return 1;
if (starts_with(url, "http://") ||
starts_with(url, "https://") ||
starts_with(url, "ftp://") ||
starts_with(url, "ftps://")) {
*out = url;
return 1;
}
return 0;
}
static int check_submodule_url(const char *url) static int check_submodule_url(const char *url)
{ {
struct credential c = CREDENTIAL_INIT; const char *curl_url;
int ret;
if (looks_like_command_line_option(url)) if (looks_like_command_line_option(url))
return -1; return -1;
ret = credential_from_url_gently(&c, url, 1); if (submodule_url_is_relative(url)) {
credential_clear(&c); char *decoded;
return ret; const char *next;
int has_nl;
/*
* This could be appended to an http URL and url-decoded;
* check for malicious characters.
*/
decoded = url_decode(url);
has_nl = !!strchr(decoded, '\n');
free(decoded);
if (has_nl)
return -1;
/*
* URLs which escape their root via "../" can overwrite
* the host field and previous components, resolving to
* URLs like https::example.com/submodule.git and
* https:///example.com/submodule.git that were
* susceptible to CVE-2020-11008.
*/
if (count_leading_dotdots(url, &next) > 0 &&
(*next == ':' || *next == '/'))
return -1;
}
else if (url_to_curl_url(url, &curl_url)) {
struct credential c = CREDENTIAL_INIT;
int ret = 0;
if (credential_from_url_gently(&c, curl_url, 1) ||
!*c.host)
ret = -1;
credential_clear(&c);
return ret;
}
return 0;
} }
struct fsck_gitmodules_data { struct fsck_gitmodules_data {
......
...@@ -558,6 +558,7 @@ static int has_cert_password(void) ...@@ -558,6 +558,7 @@ static int has_cert_password(void)
return 0; return 0;
if (!cert_auth.password) { if (!cert_auth.password) {
cert_auth.protocol = xstrdup("cert"); cert_auth.protocol = xstrdup("cert");
cert_auth.host = xstrdup("");
cert_auth.username = xstrdup(""); cert_auth.username = xstrdup("");
cert_auth.path = xstrdup(ssl_cert); cert_auth.path = xstrdup(ssl_cert);
credential_fill(&cert_auth); credential_fill(&cert_auth);
......
...@@ -22,6 +22,11 @@ test_expect_success 'setup helper scripts' ' ...@@ -22,6 +22,11 @@ test_expect_success 'setup helper scripts' '
exit 0 exit 0
EOF EOF
write_script git-credential-quit <<-\EOF &&
. ./dump
echo quit=1
EOF
write_script git-credential-verbatim <<-\EOF && write_script git-credential-verbatim <<-\EOF &&
user=$1; shift user=$1; shift
pass=$1; shift pass=$1; shift
...@@ -35,43 +40,71 @@ test_expect_success 'setup helper scripts' ' ...@@ -35,43 +40,71 @@ test_expect_success 'setup helper scripts' '
test_expect_success 'credential_fill invokes helper' ' test_expect_success 'credential_fill invokes helper' '
check fill "verbatim foo bar" <<-\EOF check fill "verbatim foo bar" <<-\EOF
protocol=http
host=example.com
-- --
protocol=http
host=example.com
username=foo username=foo
password=bar password=bar
-- --
verbatim: get verbatim: get
verbatim: protocol=http
verbatim: host=example.com
EOF EOF
' '
test_expect_success 'credential_fill invokes multiple helpers' ' test_expect_success 'credential_fill invokes multiple helpers' '
check fill useless "verbatim foo bar" <<-\EOF check fill useless "verbatim foo bar" <<-\EOF
protocol=http
host=example.com
-- --
protocol=http
host=example.com
username=foo username=foo
password=bar password=bar
-- --
useless: get useless: get
useless: protocol=http
useless: host=example.com
verbatim: get verbatim: get
verbatim: protocol=http
verbatim: host=example.com
EOF EOF
' '
test_expect_success 'credential_fill stops when we get a full response' ' test_expect_success 'credential_fill stops when we get a full response' '
check fill "verbatim one two" "verbatim three four" <<-\EOF check fill "verbatim one two" "verbatim three four" <<-\EOF
protocol=http
host=example.com
-- --
protocol=http
host=example.com
username=one username=one
password=two password=two
-- --
verbatim: get verbatim: get
verbatim: protocol=http
verbatim: host=example.com
EOF EOF
' '
test_expect_success 'credential_fill continues through partial response' ' test_expect_success 'credential_fill continues through partial response' '
check fill "verbatim one \"\"" "verbatim two three" <<-\EOF check fill "verbatim one \"\"" "verbatim two three" <<-\EOF
protocol=http
host=example.com
-- --
protocol=http
host=example.com
username=two username=two
password=three password=three
-- --
verbatim: get verbatim: get
verbatim: protocol=http
verbatim: host=example.com
verbatim: get verbatim: get
verbatim: protocol=http
verbatim: host=example.com
verbatim: username=one verbatim: username=one
EOF EOF
' '
...@@ -97,14 +130,20 @@ test_expect_success 'credential_fill passes along metadata' ' ...@@ -97,14 +130,20 @@ test_expect_success 'credential_fill passes along metadata' '
test_expect_success 'credential_approve calls all helpers' ' test_expect_success 'credential_approve calls all helpers' '
check approve useless "verbatim one two" <<-\EOF check approve useless "verbatim one two" <<-\EOF
protocol=http
host=example.com
username=foo username=foo
password=bar password=bar
-- --
-- --
useless: store useless: store
useless: protocol=http
useless: host=example.com
useless: username=foo useless: username=foo
useless: password=bar useless: password=bar
verbatim: store verbatim: store
verbatim: protocol=http
verbatim: host=example.com
verbatim: username=foo verbatim: username=foo
verbatim: password=bar verbatim: password=bar
EOF EOF
...@@ -112,6 +151,8 @@ test_expect_success 'credential_approve calls all helpers' ' ...@@ -112,6 +151,8 @@ test_expect_success 'credential_approve calls all helpers' '
test_expect_success 'do not bother storing password-less credential' ' test_expect_success 'do not bother storing password-less credential' '
check approve useless <<-\EOF check approve useless <<-\EOF
protocol=http
host=example.com
username=foo username=foo
-- --
-- --
...@@ -121,14 +162,20 @@ test_expect_success 'do not bother storing password-less credential' ' ...@@ -121,14 +162,20 @@ test_expect_success 'do not bother storing password-less credential' '
test_expect_success 'credential_reject calls all helpers' ' test_expect_success 'credential_reject calls all helpers' '
check reject useless "verbatim one two" <<-\EOF check reject useless "verbatim one two" <<-\EOF
protocol=http
host=example.com
username=foo username=foo
password=bar password=bar
-- --
-- --
useless: erase useless: erase
useless: protocol=http
useless: host=example.com
useless: username=foo useless: username=foo
useless: password=bar useless: password=bar
verbatim: erase verbatim: erase
verbatim: protocol=http
verbatim: host=example.com
verbatim: username=foo verbatim: username=foo
verbatim: password=bar verbatim: password=bar
EOF EOF
...@@ -136,33 +183,49 @@ test_expect_success 'credential_reject calls all helpers' ' ...@@ -136,33 +183,49 @@ test_expect_success 'credential_reject calls all helpers' '
test_expect_success 'usernames can be preserved' ' test_expect_success 'usernames can be preserved' '
check fill "verbatim \"\" three" <<-\EOF check fill "verbatim \"\" three" <<-\EOF
protocol=http
host=example.com
username=one username=one
-- --
protocol=http
host=example.com
username=one username=one
password=three password=three
-- --
verbatim: get verbatim: get
verbatim: protocol=http
verbatim: host=example.com
verbatim: username=one verbatim: username=one
EOF EOF
' '
test_expect_success 'usernames can be overridden' ' test_expect_success 'usernames can be overridden' '
check fill "verbatim two three" <<-\EOF check fill "verbatim two three" <<-\EOF
protocol=http
host=example.com
username=one username=one
-- --
protocol=http
host=example.com
username=two username=two
password=three password=three
-- --
verbatim: get verbatim: get
verbatim: protocol=http
verbatim: host=example.com
verbatim: username=one verbatim: username=one
EOF EOF
' '
test_expect_success 'do not bother completing already-full credential' ' test_expect_success 'do not bother completing already-full credential' '
check fill "verbatim three four" <<-\EOF check fill "verbatim three four" <<-\EOF
protocol=http
host=example.com
username=one username=one
password=two password=two
-- --
protocol=http
host=example.com
username=one username=one
password=two password=two
-- --
...@@ -174,23 +237,31 @@ test_expect_success 'do not bother completing already-full credential' ' ...@@ -174,23 +237,31 @@ test_expect_success 'do not bother completing already-full credential' '
# askpass helper is run, we know the internal getpass is working. # askpass helper is run, we know the internal getpass is working.
test_expect_success 'empty helper list falls back to internal getpass' ' test_expect_success 'empty helper list falls back to internal getpass' '
check fill <<-\EOF check fill <<-\EOF
protocol=http
host=example.com
-- --
protocol=http
host=example.com
username=askpass-username username=askpass-username
password=askpass-password password=askpass-password
-- --
askpass: Username: askpass: Username for '\''http://example.com'\'':
askpass: Password: askpass: Password for '\''http://askpass-username@example.com'\'':
EOF EOF
' '
test_expect_success 'internal getpass does not ask for known username' ' test_expect_success 'internal getpass does not ask for known username' '
check fill <<-\EOF check fill <<-\EOF
protocol=http
host=example.com
username=foo username=foo
-- --
protocol=http
host=example.com
username=foo username=foo
password=askpass-password password=askpass-password
-- --
askpass: Password: askpass: Password for '\''http://foo@example.com'\'':
EOF EOF
' '
...@@ -202,7 +273,11 @@ HELPER="!f() { ...@@ -202,7 +273,11 @@ HELPER="!f() {
test_expect_success 'respect configured credentials' ' test_expect_success 'respect configured credentials' '
test_config credential.helper "$HELPER" && test_config credential.helper "$HELPER" &&
check fill <<-\EOF check fill <<-\EOF
protocol=http
host=example.com
-- --
protocol=http
host=example.com
username=foo username=foo
password=bar password=bar
-- --
...@@ -291,35 +366,85 @@ test_expect_success 'http paths can be part of context' ' ...@@ -291,35 +366,85 @@ test_expect_success 'http paths can be part of context' '
test_expect_success 'helpers can abort the process' ' test_expect_success 'helpers can abort the process' '
test_must_fail git \ test_must_fail git \
-c credential.helper="!f() { echo quit=1; }; f" \ -c credential.helper=quit \
-c credential.helper="verbatim foo bar" \ -c credential.helper="verbatim foo bar" \
credential fill >stdout && credential fill >stdout 2>stderr <<-\EOF &&
test_must_be_empty stdout protocol=http
host=example.com
EOF
test_must_be_empty stdout &&
cat >expect <<-\EOF &&
quit: get
quit: protocol=http
quit: host=example.com
fatal: credential helper '\''quit'\'' told us to quit
EOF
test_i18ncmp expect stderr
' '
test_expect_success 'empty helper spec resets helper list' ' test_expect_success 'empty helper spec resets helper list' '
test_config credential.helper "verbatim file file" && test_config credential.helper "verbatim file file" &&
check fill "" "verbatim cmdline cmdline" <<-\EOF check fill "" "verbatim cmdline cmdline" <<-\EOF
protocol=http
host=example.com
-- --
protocol=http
host=example.com
username=cmdline username=cmdline
password=cmdline password=cmdline
-- --
verbatim: get verbatim: get
verbatim: protocol=http
verbatim: host=example.com
EOF EOF
' '
test_expect_success 'url parser ignores embedded newlines' ' test_expect_success 'url parser rejects embedded newlines' '
check fill <<-EOF test_must_fail git credential fill 2>stderr <<-\EOF &&
url=https://one.example.com?%0ahost=two.example.com/ url=https://one.example.com?%0ahost=two.example.com/
EOF
cat >expect <<-\EOF &&
warning: url contains a newline in its host component: https://one.example.com?%0ahost=two.example.com/
fatal: credential url cannot be parsed: https://one.example.com?%0ahost=two.example.com/
EOF
test_i18ncmp expect stderr
'
test_expect_success 'host-less URLs are parsed as empty host' '
check fill "verbatim foo bar" <<-\EOF
url=cert:///path/to/cert.pem
-- --
username=askpass-username protocol=cert
password=askpass-password host=
path=path/to/cert.pem
username=foo
password=bar
-- --
warning: url contains a newline in its host component: https://one.example.com?%0ahost=two.example.com/ verbatim: get
warning: skipping credential lookup for url: https://one.example.com?%0ahost=two.example.com/ verbatim: protocol=cert
askpass: Username: verbatim: host=
askpass: Password: verbatim: path=path/to/cert.pem
EOF
'
test_expect_success 'credential system refuses to work with missing host' '
test_must_fail git credential fill 2>stderr <<-\EOF &&
protocol=http
EOF
cat >expect <<-\EOF &&
fatal: refusing to work with credential missing host field
EOF
test_i18ncmp expect stderr
'
test_expect_success 'credential system refuses to work with missing protocol' '
test_must_fail git credential fill 2>stderr <<-\EOF &&
host=example.com
EOF
cat >expect <<-\EOF &&
fatal: refusing to work with credential missing protocol field
EOF EOF
test_i18ncmp expect stderr
' '
test_done test_done
...@@ -321,11 +321,17 @@ test_expect_success 'git client does not send an empty Accept-Language' ' ...@@ -321,11 +321,17 @@ test_expect_success 'git client does not send an empty Accept-Language' '
' '
test_expect_success 'remote-http complains cleanly about malformed urls' ' test_expect_success 'remote-http complains cleanly about malformed urls' '
# do not actually issue "list" or other commands, as we do not test_must_fail git remote-http http::/example.com/repo.git 2>stderr &&
# want to rely on what curl would actually do with such a broken test_i18ngrep "url has no scheme" stderr
# URL. This is just about making sure we do not segfault during '
# initialization.
test_must_fail git remote-http http::/example.com/repo.git # NEEDSWORK: Writing commands to git-remote-curl can race against the latter
# erroring out, producing SIGPIPE. Remove "ok=sigpipe" once transport-helper has
# learned to handle early remote helper failures more cleanly.
test_expect_success 'remote-http complains cleanly about empty scheme' '
test_must_fail ok=sigpipe git ls-remote \
http::${HTTPD_URL#http}/dumb/repo.git 2>stderr &&
test_i18ngrep "url has no scheme" stderr
' '
test_expect_success 'redirects can be forbidden/allowed' ' test_expect_success 'redirects can be forbidden/allowed' '
......
...@@ -60,6 +60,116 @@ test_expect_success 'trailing backslash is handled correctly' ' ...@@ -60,6 +60,116 @@ test_expect_success 'trailing backslash is handled correctly' '
test_i18ngrep ! "unknown option" err test_i18ngrep ! "unknown option" err
' '
test_expect_success 'fsck rejects missing URL scheme' '
git checkout --orphan missing-scheme &&
cat >.gitmodules <<-\EOF &&
[submodule "foo"]
url = http::one.example.com/foo.git
EOF
git add .gitmodules &&
test_tick &&
git commit -m "gitmodules with missing URL scheme" &&
test_when_finished "rm -rf dst" &&
git init --bare dst &&
git -C dst config transfer.fsckObjects true &&
test_must_fail git push dst HEAD 2>err &&
grep gitmodulesUrl err
'
test_expect_success 'fsck rejects relative URL resolving to missing scheme' '
git checkout --orphan relative-missing-scheme &&
cat >.gitmodules <<-\EOF &&
[submodule "foo"]
url = "..\\../.\\../:one.example.com/foo.git"
EOF
git add .gitmodules &&
test_tick &&
git commit -m "gitmodules with relative URL that strips off scheme" &&
test_when_finished "rm -rf dst" &&
git init --bare dst &&
git -C dst config transfer.fsckObjects true &&
test_must_fail git push dst HEAD 2>err &&
grep gitmodulesUrl err
'
test_expect_success 'fsck rejects empty URL scheme' '
git checkout --orphan empty-scheme &&
cat >.gitmodules <<-\EOF &&
[submodule "foo"]
url = http::://one.example.com/foo.git
EOF
git add .gitmodules &&
test_tick &&
git commit -m "gitmodules with empty URL scheme" &&
test_when_finished "rm -rf dst" &&
git init --bare dst &&
git -C dst config transfer.fsckObjects true &&
test_must_fail git push dst HEAD 2>err &&
grep gitmodulesUrl err
'
test_expect_success 'fsck rejects relative URL resolving to empty scheme' '
git checkout --orphan relative-empty-scheme &&
cat >.gitmodules <<-\EOF &&
[submodule "foo"]
url = ../../../:://one.example.com/foo.git
EOF
git add .gitmodules &&
test_tick &&
git commit -m "relative gitmodules URL resolving to empty scheme" &&
test_when_finished "rm -rf dst" &&
git init --bare dst &&
git -C dst config transfer.fsckObjects true &&
test_must_fail git push dst HEAD 2>err &&
grep gitmodulesUrl err
'
test_expect_success 'fsck rejects empty hostname' '
git checkout --orphan empty-host &&
cat >.gitmodules <<-\EOF &&
[submodule "foo"]
url = http:///one.example.com/foo.git
EOF
git add .gitmodules &&
test_tick &&
git commit -m "gitmodules with extra slashes" &&
test_when_finished "rm -rf dst" &&
git init --bare dst &&
git -C dst config transfer.fsckObjects true &&
test_must_fail git push dst HEAD 2>err &&
grep gitmodulesUrl err
'
test_expect_success 'fsck rejects relative url that produced empty hostname' '
git checkout --orphan messy-relative &&
cat >.gitmodules <<-\EOF &&
[submodule "foo"]
url = ../../..//one.example.com/foo.git
EOF
git add .gitmodules &&
test_tick &&
git commit -m "gitmodules abusing relative_path" &&
test_when_finished "rm -rf dst" &&
git init --bare dst &&
git -C dst config transfer.fsckObjects true &&
test_must_fail git push dst HEAD 2>err &&
grep gitmodulesUrl err
'
test_expect_success 'fsck permits embedded newline with unrecognized scheme' '
git checkout --orphan newscheme &&
cat >.gitmodules <<-\EOF &&
[submodule "foo"]
url = "data://acjbkd%0akajfdickajkd"
EOF
git add .gitmodules &&
git commit -m "gitmodules with unrecognized scheme" &&
test_when_finished "rm -rf dst" &&
git init --bare dst &&
git -C dst config transfer.fsckObjects true &&
git push dst HEAD
'
test_expect_success 'fsck rejects embedded newline in url' ' test_expect_success 'fsck rejects embedded newline in url' '
# create an orphan branch to avoid existing .gitmodules objects # create an orphan branch to avoid existing .gitmodules objects
git checkout --orphan newline && git checkout --orphan newline &&
...@@ -76,4 +186,19 @@ test_expect_success 'fsck rejects embedded newline in url' ' ...@@ -76,4 +186,19 @@ test_expect_success 'fsck rejects embedded newline in url' '
grep gitmodulesUrl err grep gitmodulesUrl err
' '
test_expect_success 'fsck rejects embedded newline in relative url' '
git checkout --orphan relative-newline &&
cat >.gitmodules <<-\EOF &&
[submodule "foo"]
url = "./%0ahost=two.example.com/foo.git"
EOF
git add .gitmodules &&
git commit -m "relative url with newline" &&
test_when_finished "rm -rf dst" &&
git init --bare dst &&
git -C dst config transfer.fsckObjects true &&
test_must_fail git push dst HEAD 2>err &&
grep gitmodulesUrl err
'
test_done test_done
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册