"git checkout -- <paths>" is usually used to restore all modified
files in <paths>. In sparse checkout mode, this command is overloaded
with another meaning: to add back all files in <paths> that are
excluded by sparse patterns.

Add "--no-widen" option to do what normal mode does: restore all
modified files and nothing else. The user can make it "default" by
putting it to an alias (e.g. "co = checkout --no-widen") and get back
to the original behavior with "git co --widen".

Signed-off-by: Nguyễn Thái Ngọc Duy <pclo...@gmail.com>
---
 --sparse -> --no-widen and git-checkout does not error out when the
 option is specified but has no effects. Let's see how it works out.
 We could change the default from --widen to --no-widen later.

 Documentation/git-checkout.txt       | 11 +++++++++++
 builtin/checkout.c                   |  7 +++++++
 t/t1011-read-tree-sparse-checkout.sh | 12 ++++++++++++
 3 files changed, 30 insertions(+)

diff --git a/Documentation/git-checkout.txt b/Documentation/git-checkout.txt
index 8edcdca..538b8b8 100644
--- a/Documentation/git-checkout.txt
+++ b/Documentation/git-checkout.txt
@@ -180,6 +180,17 @@ branch by running "git rm -rf ." from the top level of the 
working tree.
 Afterwards you will be ready to prepare your new files, repopulating the
 working tree, by copying them from elsewhere, extracting a tarball, etc.
 
+--no-widen::
+       In sparse checkout mode, `git checkout -- <paths>` would
+       update all entries matched by <paths> regardless of sparse
+       patterns. This option only updates entries matched by <paths>
+       and sparse patterns.
+
+--widen::
+       Revert the effect of `--no-widen` if specified and make
+       `git checkout -- <paths>` update all entries matched by
+       <paths> regardless of sparse patterns.
+
 -m::
 --merge::
        When switching branches,
diff --git a/builtin/checkout.c b/builtin/checkout.c
index a9c1b5a..246de33 100644
--- a/builtin/checkout.c
+++ b/builtin/checkout.c
@@ -35,6 +35,7 @@ struct checkout_opts {
        int force_detach;
        int writeout_stage;
        int overwrite_ignore;
+       int widen_sparse;
 
        const char *new_branch;
        const char *new_branch_force;
@@ -275,6 +276,8 @@ static int checkout_paths(const struct checkout_opts *opts,
                struct cache_entry *ce = active_cache[pos];
                if (opts->source_tree && !(ce->ce_flags & CE_UPDATE))
                        continue;
+               if (!opts->widen_sparse && ce_skip_worktree(ce))
+                       continue;
                match_pathspec(opts->pathspec, ce->name, ce_namelen(ce), 0, 
ps_matched);
        }
 
@@ -315,6 +318,8 @@ static int checkout_paths(const struct checkout_opts *opts,
                struct cache_entry *ce = active_cache[pos];
                if (opts->source_tree && !(ce->ce_flags & CE_UPDATE))
                        continue;
+               if (!opts->widen_sparse && ce_skip_worktree(ce))
+                       continue;
                if (match_pathspec(opts->pathspec, ce->name, ce_namelen(ce), 0, 
NULL)) {
                        if (!ce_stage(ce)) {
                                errs |= checkout_entry(ce, &state, NULL);
@@ -1029,6 +1034,7 @@ int cmd_checkout(int argc, const char **argv, const char 
*prefix)
                OPT_STRING(0, "conflict", &conflict_style, N_("style"),
                           N_("conflict style (merge or diff3)")),
                OPT_BOOLEAN('p', "patch", &opts.patch_mode, N_("select hunks 
interactively")),
+               OPT_BOOL(0, "widen", &opts.widen_sparse, N_("do not limit 
pathspecs to sparse entries only (default)")),
                { OPTION_BOOLEAN, 0, "guess", &dwim_new_local_branch, NULL,
                  N_("second guess 'git checkout no-such-branch'"),
                  PARSE_OPT_NOARG | PARSE_OPT_HIDDEN },
@@ -1039,6 +1045,7 @@ int cmd_checkout(int argc, const char **argv, const char 
*prefix)
        memset(&new, 0, sizeof(new));
        opts.overwrite_ignore = 1;
        opts.prefix = prefix;
+       opts.widen_sparse = 1;
 
        gitmodules_config();
        git_config(git_checkout_config, &opts);
diff --git a/t/t1011-read-tree-sparse-checkout.sh 
b/t/t1011-read-tree-sparse-checkout.sh
index 5c0053a..4a45714 100755
--- a/t/t1011-read-tree-sparse-checkout.sh
+++ b/t/t1011-read-tree-sparse-checkout.sh
@@ -250,4 +250,16 @@ EOF
        test_cmp expected actual
 '
 
+test_expect_success 'checkout --no-widen' '
+       echo "*" >.git/info/sparse-checkout &&
+       git checkout -f top &&
+       test -f init.t &&
+       echo sub >.git/info/sparse-checkout &&
+       git checkout &&
+       echo modified >> sub/added &&
+       git checkout --no-widen . &&
+       ! test -f init.t &&
+       git diff --exit-code HEAD
+'
+
 test_done
-- 
1.8.2.83.gc99314b

--
To unsubscribe from this list: send the line "unsubscribe git" in
the body of a message to majord...@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html

Reply via email to