Merge branch 'nd/attr-pathspec-fix'
authorJunio C Hamano <gitster@pobox.com>
Mon, 24 Sep 2018 17:30:51 +0000 (10:30 -0700)
committerJunio C Hamano <gitster@pobox.com>
Mon, 24 Sep 2018 17:30:51 +0000 (10:30 -0700)
"git add ':(attr:foo)'" is not supported and is supposed to be
rejected while the command line arguments are parsed, but we fail
to reject such a command line upfront.

* nd/attr-pathspec-fix:
add: do not accept pathspec magic 'attr'

1  2 
builtin/add.c
diff --combined builtin/add.c
index 9916498a29bbd8fa7c5c5d8e7bd32e1dc184909b,cb1d961508a3a19e447c06f253b3c1385ec1d35e..0b64bcdebe0f4581953adccf8addf1c3fd1eb61e
@@@ -9,7 -9,7 +9,7 @@@
  #include "lockfile.h"
  #include "dir.h"
  #include "pathspec.h"
 -#include "exec_cmd.h"
 +#include "exec-cmd.h"
  #include "cache-tree.h"
  #include "run-command.h"
  #include "parse-options.h"
@@@ -26,7 -26,6 +26,7 @@@ static const char * const builtin_add_u
  };
  static int patch_interactive, add_interactive, edit_interactive;
  static int take_worktree_changes;
 +static int add_renormalize;
  
  struct update_callback_data {
        int flags;
@@@ -40,7 -39,7 +40,7 @@@ static void chmod_pathspec(struct paths
        for (i = 0; i < active_nr; i++) {
                struct cache_entry *ce = active_cache[i];
  
 -              if (pathspec && !ce_path_match(ce, pathspec, NULL))
 +              if (pathspec && !ce_path_match(&the_index, ce, pathspec, NULL))
                        continue;
  
                if (chmod_cache_entry(ce, flip) < 0)
@@@ -117,32 -116,13 +117,32 @@@ int add_files_to_cache(const char *pref
        rev.diffopt.output_format = DIFF_FORMAT_CALLBACK;
        rev.diffopt.format_callback = update_callback;
        rev.diffopt.format_callback_data = &data;
 -      rev.diffopt.flags |= DIFF_OPT_OVERRIDE_SUBMODULE_CONFIG;
 +      rev.diffopt.flags.override_submodule_config = 1;
        rev.max_count = 0; /* do not compare unmerged paths with stage #2 */
        run_diff_files(&rev, DIFF_RACY_IS_MODIFIED);
        clear_pathspec(&rev.prune_data);
        return !!data.add_errors;
  }
  
 +static int renormalize_tracked_files(const struct pathspec *pathspec, int flags)
 +{
 +      int i, retval = 0;
 +
 +      for (i = 0; i < active_nr; i++) {
 +              struct cache_entry *ce = active_cache[i];
 +
 +              if (ce_stage(ce))
 +                      continue; /* do not touch unmerged paths */
 +              if (!S_ISREG(ce->ce_mode) && !S_ISLNK(ce->ce_mode))
 +                      continue; /* do not touch non blobs */
 +              if (pathspec && !ce_path_match(&the_index, ce, pathspec, NULL))
 +                      continue;
 +              retval |= add_file_to_cache(ce->name, flags | HASH_RENORMALIZE);
 +      }
 +
 +      return retval;
 +}
 +
  static char *prune_directory(struct dir_struct *dir, struct pathspec *pathspec, int prefix)
  {
        char *seen;
        i = dir->nr;
        while (--i >= 0) {
                struct dir_entry *entry = *src++;
 -              if (dir_path_match(entry, pathspec, prefix, seen))
 +              if (dir_path_match(&the_index, entry, pathspec, prefix, seen))
                        *dst++ = entry;
        }
        dir->nr = dst - dir->entries;
@@@ -238,7 -218,7 +238,7 @@@ static int edit_patch(int argc, const c
        argc = setup_revisions(argc, argv, &rev, NULL);
        rev.diffopt.output_format = DIFF_FORMAT_PATCH;
        rev.diffopt.use_color = 0;
 -      DIFF_OPT_SET(&rev.diffopt, IGNORE_DIRTY_SUBMODULES);
 +      rev.diffopt.flags.ignore_dirty_submodules = 1;
        out = open(file, O_CREAT | O_WRONLY, 0666);
        if (out < 0)
                die(_("Could not open '%s' for writing."), file);
        return 0;
  }
  
 -static struct lock_file lock_file;
 -
  static const char ignore_error[] =
  N_("The following paths are ignored by one of your .gitignore files:\n");
  
@@@ -292,9 -274,8 +292,9 @@@ static struct option builtin_add_option
        OPT_BOOL('i', "interactive", &add_interactive, N_("interactive picking")),
        OPT_BOOL('p', "patch", &patch_interactive, N_("select hunks interactively")),
        OPT_BOOL('e', "edit", &edit_interactive, N_("edit current diff and apply")),
 -      OPT__FORCE(&ignored_too, N_("allow adding otherwise ignored files")),
 +      OPT__FORCE(&ignored_too, N_("allow adding otherwise ignored files"), 0),
        OPT_BOOL('u', "update", &take_worktree_changes, N_("update tracked files")),
 +      OPT_BOOL(0, "renormalize", &add_renormalize, N_("renormalize EOL of tracked files (implies -u)")),
        OPT_BOOL('N', "intent-to-add", &intent_to_add, N_("record only the fact that the path will be added later")),
        OPT_BOOL('A', "all", &addremove_explicit, N_("add changes from all tracked and untracked files")),
        { OPTION_CALLBACK, 0, "ignore-removal", &addremove_explicit,
        OPT_BOOL( 0 , "refresh", &refresh_only, N_("don't add, only refresh the index")),
        OPT_BOOL( 0 , "ignore-errors", &ignore_add_errors, N_("just skip files which cannot be added because of errors")),
        OPT_BOOL( 0 , "ignore-missing", &ignore_missing, N_("check if - even missing - files are ignored in dry run")),
 -      OPT_STRING( 0 , "chmod", &chmod_arg, N_("(+/-)x"), N_("override the executable bit of the listed files")),
 +      OPT_STRING(0, "chmod", &chmod_arg, "(+|-)x",
 +                 N_("override the executable bit of the listed files")),
        OPT_HIDDEN_BOOL(0, "warn-embedded-repo", &warn_on_embedded_repo,
                        N_("warn when adding an embedded repository")),
        OPT_END(),
@@@ -392,7 -372,6 +392,7 @@@ int cmd_add(int argc, const char **argv
        int add_new_files;
        int require_pathspec;
        char *seen = NULL;
 +      struct lock_file lock_file = LOCK_INIT;
  
        git_config(add_config, NULL);
  
                          chmod_arg[1] != 'x' || chmod_arg[2]))
                die(_("--chmod param '%s' must be either -x or +x"), chmod_arg);
  
 -      add_new_files = !take_worktree_changes && !refresh_only;
 +      add_new_files = !take_worktree_changes && !refresh_only && !add_renormalize;
        require_pathspec = !(take_worktree_changes || (0 < addremove_explicit));
  
        hold_locked_index(&lock_file, LOCK_DIE_ON_ERROR);
         * Check the "pathspec '%s' did not match any files" block
         * below before enabling new magic.
         */
-       parse_pathspec(&pathspec, 0,
+       parse_pathspec(&pathspec, PATHSPEC_ATTR,
                       PATHSPEC_PREFER_FULL |
                       PATHSPEC_SYMLINK_LEADING_PATH,
                       prefix, argv);
  
        plug_bulk_checkin();
  
 -      exit_status |= add_files_to_cache(prefix, &pathspec, flags);
 +      if (add_renormalize)
 +              exit_status |= renormalize_tracked_files(&pathspec, flags);
 +      else
 +              exit_status |= add_files_to_cache(prefix, &pathspec, flags);
  
        if (add_new_files)
                exit_status |= add_files(&dir, flags);
        unplug_bulk_checkin();
  
  finish:
 -      if (active_cache_changed) {
 -              if (write_locked_index(&the_index, &lock_file, COMMIT_LOCK))
 -                      die(_("Unable to write new index file"));
 -      }
 +      if (write_locked_index(&the_index, &lock_file,
 +                             COMMIT_LOCK | SKIP_IF_UNCHANGED))
 +              die(_("Unable to write new index file"));
  
        UNLEAK(pathspec);
        UNLEAK(dir);