Message ID | pull.1488.v3.git.git.1681232918484.gitgitgadget@gmail.com (mailing list archive) |
---|---|
State | New, archived |
Headers | show |
Series | [v3] clone: error specifically with --local and symlinked objects | expand |
"Glen Choo via GitGitGadget" <gitgitgadget@gmail.com> writes: > A quick reroll to fix the style issue Taylor spotted in v2 (good > catch!). I didn't spot this patch in 'next' or 'seen', so hopefully I'm > still in time :) What I locally have, which is an amended version of v2, and this one differ like so: diff --git a/builtin/clone.c b/builtin/clone.c index ae2db8535a..b42231758c 100644 --- a/builtin/clone.c +++ b/builtin/clone.c @@ -335,7 +335,6 @@ static void copy_or_link_directory(struct strbuf *src, struct strbuf *dest, if (errno == ENOTDIR) { int saved_errno = errno; struct stat st; - if (!lstat(src->buf, &st) && S_ISLNK(st.st_mode)) die(_("'%s' is a symlink, refusing to clone with --local"), src->buf); So I'll keep my copy. Thanks anyway.
Junio C Hamano <gitster@pobox.com> writes: > What I locally have, which is an amended version of v2, and this one > differ like so: > > diff --git a/builtin/clone.c b/builtin/clone.c > index ae2db8535a..b42231758c 100644 > --- a/builtin/clone.c > +++ b/builtin/clone.c > @@ -335,7 +335,6 @@ static void copy_or_link_directory(struct strbuf *src, struct strbuf *dest, > if (errno == ENOTDIR) { > int saved_errno = errno; > struct stat st; > - > if (!lstat(src->buf, &st) && S_ISLNK(st.st_mode)) > die(_("'%s' is a symlink, refusing to clone with --local"), > src->buf); > > So I'll keep my copy. Ah, sharp eye. Sounds good.
diff --git a/Documentation/git-clone.txt b/Documentation/git-clone.txt index d6434d262d6..c37c4a37f74 100644 --- a/Documentation/git-clone.txt +++ b/Documentation/git-clone.txt @@ -58,6 +58,11 @@ never use the local optimizations). Specifying `--no-local` will override the default when `/path/to/repo` is given, using the regular Git transport instead. + +If the repository's `$GIT_DIR/objects` has symbolic links or is a +symbolic link, the clone will fail. This is a security measure to +prevent the unintentional copying of files by dereferencing the symbolic +links. ++ *NOTE*: this operation can race with concurrent modification to the source repository, similar to running `cp -r src dst` while modifying `src`. diff --git a/builtin/clone.c b/builtin/clone.c index 462c286274c..dd2b4e130e6 100644 --- a/builtin/clone.c +++ b/builtin/clone.c @@ -327,8 +327,17 @@ static void copy_or_link_directory(struct strbuf *src, struct strbuf *dest, iter = dir_iterator_begin(src->buf, DIR_ITERATOR_PEDANTIC); - if (!iter) + if (!iter) { + if (errno == ENOTDIR) { + int saved_errno = errno; + struct stat st; + if (!lstat(src->buf, &st) && S_ISLNK(st.st_mode)) + die(_("'%s' is a symlink, refusing to clone with --local"), + src->buf); + errno = saved_errno; + } die_errno(_("failed to start iterator over '%s'"), src->buf); + } strbuf_addch(src, '/'); src_len = src->len; diff --git a/t/t5604-clone-reference.sh b/t/t5604-clone-reference.sh index 83e3c97861d..9845fc04d59 100755 --- a/t/t5604-clone-reference.sh +++ b/t/t5604-clone-reference.sh @@ -358,7 +358,7 @@ test_expect_success SYMLINKS 'clone repo with symlinked objects directory' ' test_must_fail git clone --local malicious clone 2>err && test_path_is_missing clone && - grep "failed to start iterator over" err + grep "is a symlink, refusing to clone with --local" err ' test_done