зеркало из https://github.com/microsoft/git.git
status: add option to show ignored files differently
Teach the status command more flexibility in how ignored files are reported. Currently, the reporting of ignored files and untracked files are linked. You cannot control how ignored files are reported independently of how untracked files are reported (i.e. `all` vs `normal`). This makes it impossible to show untracked files with the `all` option, but show ignored files with the `normal` option. This work 1) adds the ability to control the reporting of ignored files independently of untracked files and 2) introduces the concept of status reporting ignored paths that explicitly match an ignored pattern. There are 2 benefits to these changes: 1) if a consumer needs all untracked files but not all ignored files, there is a performance benefit to not scanning all contents of an ignored directory and 2) returning ignored files that explicitly match a path allow a consumer to make more informed decisions about when a status result might be stale. This commit implements --ignored=matching with --untracked-files=all. The following commit will implement --ignored=matching with --untracked=files=normal. As an example of where this flexibility could be useful is that our application (Visual Studio) runs the status command and presents the output. It shows all untracked files individually (e.g. using the '--untracked-files==all' option), and would like to know about which paths are ignored. It uses information about ignored paths to make decisions about when the status result might have changed. Additionally, many projects place build output into directories inside a repository's working directory (e.g. in "bin/" and "obj/" directories). Normal usage is to explicitly ignore these 2 directory names in the .gitignore file (rather than or in addition to the *.obj pattern).If an application could know that these directories are explicitly ignored, it could infer that all contents are ignored as well and make better informed decisions about files in these directories. It could infer that any changes under these paths would not affect the output of status. Additionally, there can be a significant performance benefit by avoiding scanning through ignored directories. When status is set to report matching ignored files, it has the following behavior. Ignored files and directories that explicitly match an exclude pattern are reported. If an ignored directory matches an exclude pattern, then the path of the directory is returned. If a directory does not match an exclude pattern, but all of its contents are ignored, then the contained files are reported instead of the directory. Signed-off-by: Jameson Miller <jamill@microsoft.com> Signed-off-by: Junio C Hamano <gitster@pobox.com>
This commit is contained in:
Родитель
4843cdefe3
Коммит
eec0f7f2b7
|
@ -118,7 +118,7 @@ static int edit_flag = -1; /* unspecified */
|
||||||
static int quiet, verbose, no_verify, allow_empty, dry_run, renew_authorship;
|
static int quiet, verbose, no_verify, allow_empty, dry_run, renew_authorship;
|
||||||
static int config_commit_verbose = -1; /* unspecified */
|
static int config_commit_verbose = -1; /* unspecified */
|
||||||
static int no_post_rewrite, allow_empty_message;
|
static int no_post_rewrite, allow_empty_message;
|
||||||
static char *untracked_files_arg, *force_date, *ignore_submodule_arg;
|
static char *untracked_files_arg, *force_date, *ignore_submodule_arg, *ignored_arg;
|
||||||
static char *sign_commit;
|
static char *sign_commit;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -139,7 +139,7 @@ static const char *cleanup_arg;
|
||||||
static enum commit_whence whence;
|
static enum commit_whence whence;
|
||||||
static int sequencer_in_use;
|
static int sequencer_in_use;
|
||||||
static int use_editor = 1, include_status = 1;
|
static int use_editor = 1, include_status = 1;
|
||||||
static int show_ignored_in_status, have_option_m;
|
static int have_option_m;
|
||||||
static struct strbuf message = STRBUF_INIT;
|
static struct strbuf message = STRBUF_INIT;
|
||||||
|
|
||||||
static enum wt_status_format status_format = STATUS_FORMAT_UNSPECIFIED;
|
static enum wt_status_format status_format = STATUS_FORMAT_UNSPECIFIED;
|
||||||
|
@ -1075,6 +1075,19 @@ static const char *find_author_by_nickname(const char *name)
|
||||||
die(_("--author '%s' is not 'Name <email>' and matches no existing author"), name);
|
die(_("--author '%s' is not 'Name <email>' and matches no existing author"), name);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void handle_ignored_arg(struct wt_status *s)
|
||||||
|
{
|
||||||
|
if (!ignored_arg)
|
||||||
|
; /* default already initialized */
|
||||||
|
else if (!strcmp(ignored_arg, "traditional"))
|
||||||
|
s->show_ignored_mode = SHOW_TRADITIONAL_IGNORED;
|
||||||
|
else if (!strcmp(ignored_arg, "no"))
|
||||||
|
s->show_ignored_mode = SHOW_NO_IGNORED;
|
||||||
|
else if (!strcmp(ignored_arg, "matching"))
|
||||||
|
s->show_ignored_mode = SHOW_MATCHING_IGNORED;
|
||||||
|
else
|
||||||
|
die(_("Invalid ignored mode '%s'"), ignored_arg);
|
||||||
|
}
|
||||||
|
|
||||||
static void handle_untracked_files_arg(struct wt_status *s)
|
static void handle_untracked_files_arg(struct wt_status *s)
|
||||||
{
|
{
|
||||||
|
@ -1363,8 +1376,10 @@ int cmd_status(int argc, const char **argv, const char *prefix)
|
||||||
N_("mode"),
|
N_("mode"),
|
||||||
N_("show untracked files, optional modes: all, normal, no. (Default: all)"),
|
N_("show untracked files, optional modes: all, normal, no. (Default: all)"),
|
||||||
PARSE_OPT_OPTARG, NULL, (intptr_t)"all" },
|
PARSE_OPT_OPTARG, NULL, (intptr_t)"all" },
|
||||||
OPT_BOOL(0, "ignored", &show_ignored_in_status,
|
{ OPTION_STRING, 0, "ignored", &ignored_arg,
|
||||||
N_("show ignored files")),
|
N_("mode"),
|
||||||
|
N_("show ignored files, optional modes: traditional, matching, no. (Default: traditional)"),
|
||||||
|
PARSE_OPT_OPTARG, NULL, (intptr_t)"traditional" },
|
||||||
{ OPTION_STRING, 0, "ignore-submodules", &ignore_submodule_arg, N_("when"),
|
{ OPTION_STRING, 0, "ignore-submodules", &ignore_submodule_arg, N_("when"),
|
||||||
N_("ignore changes to submodules, optional when: all, dirty, untracked. (Default: all)"),
|
N_("ignore changes to submodules, optional when: all, dirty, untracked. (Default: all)"),
|
||||||
PARSE_OPT_OPTARG, NULL, (intptr_t)"all" },
|
PARSE_OPT_OPTARG, NULL, (intptr_t)"all" },
|
||||||
|
@ -1383,8 +1398,12 @@ int cmd_status(int argc, const char **argv, const char *prefix)
|
||||||
finalize_deferred_config(&s);
|
finalize_deferred_config(&s);
|
||||||
|
|
||||||
handle_untracked_files_arg(&s);
|
handle_untracked_files_arg(&s);
|
||||||
if (show_ignored_in_status)
|
handle_ignored_arg(&s);
|
||||||
s.show_ignored_files = 1;
|
|
||||||
|
if (s.show_ignored_mode == SHOW_MATCHING_IGNORED &&
|
||||||
|
s.show_untracked_files == SHOW_NO_UNTRACKED_FILES)
|
||||||
|
die(_("Unsupported combination of ignored and untracked-files arguments"));
|
||||||
|
|
||||||
parse_pathspec(&s.pathspec, 0,
|
parse_pathspec(&s.pathspec, 0,
|
||||||
PATHSPEC_PREFER_FULL,
|
PATHSPEC_PREFER_FULL,
|
||||||
prefix, argv);
|
prefix, argv);
|
||||||
|
|
24
dir.c
24
dir.c
|
@ -1389,6 +1389,30 @@ static enum path_treatment treat_directory(struct dir_struct *dir,
|
||||||
case index_nonexistent:
|
case index_nonexistent:
|
||||||
if (dir->flags & DIR_SHOW_OTHER_DIRECTORIES)
|
if (dir->flags & DIR_SHOW_OTHER_DIRECTORIES)
|
||||||
break;
|
break;
|
||||||
|
if (exclude &&
|
||||||
|
(dir->flags & DIR_SHOW_IGNORED_TOO) &&
|
||||||
|
(dir->flags & DIR_SHOW_IGNORED_TOO_MODE_MATCHING)) {
|
||||||
|
|
||||||
|
/*
|
||||||
|
* This is an excluded directory and we are
|
||||||
|
* showing ignored paths that match an exclude
|
||||||
|
* pattern. (e.g. show directory as ignored
|
||||||
|
* only if it matches an exclude pattern).
|
||||||
|
* This path will either be 'path_excluded`
|
||||||
|
* (if we are showing empty directories or if
|
||||||
|
* the directory is not empty), or will be
|
||||||
|
* 'path_none' (empty directory, and we are
|
||||||
|
* not showing empty directories).
|
||||||
|
*/
|
||||||
|
if (!(dir->flags & DIR_HIDE_EMPTY_DIRECTORIES))
|
||||||
|
return path_excluded;
|
||||||
|
|
||||||
|
if (read_directory_recursive(dir, istate, dirname, len,
|
||||||
|
untracked, 1, 1, pathspec) == path_excluded)
|
||||||
|
return path_excluded;
|
||||||
|
|
||||||
|
return path_none;
|
||||||
|
}
|
||||||
if (!(dir->flags & DIR_NO_GITLINKS)) {
|
if (!(dir->flags & DIR_NO_GITLINKS)) {
|
||||||
unsigned char sha1[20];
|
unsigned char sha1[20];
|
||||||
if (resolve_gitlink_ref(dirname, "HEAD", sha1) == 0)
|
if (resolve_gitlink_ref(dirname, "HEAD", sha1) == 0)
|
||||||
|
|
3
dir.h
3
dir.h
|
@ -152,7 +152,8 @@ struct dir_struct {
|
||||||
DIR_COLLECT_IGNORED = 1<<4,
|
DIR_COLLECT_IGNORED = 1<<4,
|
||||||
DIR_SHOW_IGNORED_TOO = 1<<5,
|
DIR_SHOW_IGNORED_TOO = 1<<5,
|
||||||
DIR_COLLECT_KILLED_ONLY = 1<<6,
|
DIR_COLLECT_KILLED_ONLY = 1<<6,
|
||||||
DIR_KEEP_UNTRACKED_CONTENTS = 1<<7
|
DIR_KEEP_UNTRACKED_CONTENTS = 1<<7,
|
||||||
|
DIR_SHOW_IGNORED_TOO_MODE_MATCHING = 1<<8
|
||||||
} flags;
|
} flags;
|
||||||
struct dir_entry **entries;
|
struct dir_entry **entries;
|
||||||
struct dir_entry **ignored;
|
struct dir_entry **ignored;
|
||||||
|
|
11
wt-status.c
11
wt-status.c
|
@ -658,10 +658,15 @@ static void wt_status_collect_untracked(struct wt_status *s)
|
||||||
if (s->show_untracked_files != SHOW_ALL_UNTRACKED_FILES)
|
if (s->show_untracked_files != SHOW_ALL_UNTRACKED_FILES)
|
||||||
dir.flags |=
|
dir.flags |=
|
||||||
DIR_SHOW_OTHER_DIRECTORIES | DIR_HIDE_EMPTY_DIRECTORIES;
|
DIR_SHOW_OTHER_DIRECTORIES | DIR_HIDE_EMPTY_DIRECTORIES;
|
||||||
if (s->show_ignored_files)
|
if (s->show_ignored_mode) {
|
||||||
dir.flags |= DIR_SHOW_IGNORED_TOO;
|
dir.flags |= DIR_SHOW_IGNORED_TOO;
|
||||||
else
|
|
||||||
|
if (s->show_ignored_mode == SHOW_MATCHING_IGNORED)
|
||||||
|
dir.flags |= DIR_SHOW_IGNORED_TOO_MODE_MATCHING;
|
||||||
|
} else {
|
||||||
dir.untracked = the_index.untracked;
|
dir.untracked = the_index.untracked;
|
||||||
|
}
|
||||||
|
|
||||||
setup_standard_excludes(&dir);
|
setup_standard_excludes(&dir);
|
||||||
|
|
||||||
fill_directory(&dir, &the_index, &s->pathspec);
|
fill_directory(&dir, &the_index, &s->pathspec);
|
||||||
|
@ -1619,7 +1624,7 @@ static void wt_longstatus_print(struct wt_status *s)
|
||||||
}
|
}
|
||||||
if (s->show_untracked_files) {
|
if (s->show_untracked_files) {
|
||||||
wt_longstatus_print_other(s, &s->untracked, _("Untracked files"), "add");
|
wt_longstatus_print_other(s, &s->untracked, _("Untracked files"), "add");
|
||||||
if (s->show_ignored_files)
|
if (s->show_ignored_mode)
|
||||||
wt_longstatus_print_other(s, &s->ignored, _("Ignored files"), "add -f");
|
wt_longstatus_print_other(s, &s->ignored, _("Ignored files"), "add -f");
|
||||||
if (advice_status_u_option && 2000 < s->untracked_in_ms) {
|
if (advice_status_u_option && 2000 < s->untracked_in_ms) {
|
||||||
status_printf_ln(s, GIT_COLOR_NORMAL, "%s", "");
|
status_printf_ln(s, GIT_COLOR_NORMAL, "%s", "");
|
||||||
|
|
|
@ -27,6 +27,12 @@ enum untracked_status_type {
|
||||||
SHOW_ALL_UNTRACKED_FILES
|
SHOW_ALL_UNTRACKED_FILES
|
||||||
};
|
};
|
||||||
|
|
||||||
|
enum show_ignored_type {
|
||||||
|
SHOW_NO_IGNORED,
|
||||||
|
SHOW_TRADITIONAL_IGNORED,
|
||||||
|
SHOW_MATCHING_IGNORED,
|
||||||
|
};
|
||||||
|
|
||||||
/* from where does this commit originate */
|
/* from where does this commit originate */
|
||||||
enum commit_whence {
|
enum commit_whence {
|
||||||
FROM_COMMIT, /* normal */
|
FROM_COMMIT, /* normal */
|
||||||
|
@ -70,7 +76,7 @@ struct wt_status {
|
||||||
int display_comment_prefix;
|
int display_comment_prefix;
|
||||||
int relative_paths;
|
int relative_paths;
|
||||||
int submodule_summary;
|
int submodule_summary;
|
||||||
int show_ignored_files;
|
enum show_ignored_type show_ignored_mode;
|
||||||
enum untracked_status_type show_untracked_files;
|
enum untracked_status_type show_untracked_files;
|
||||||
const char *ignore_submodule_arg;
|
const char *ignore_submodule_arg;
|
||||||
char color_palette[WT_STATUS_MAXSLOT][COLOR_MAXLEN];
|
char color_palette[WT_STATUS_MAXSLOT][COLOR_MAXLEN];
|
||||||
|
|
Загрузка…
Ссылка в новой задаче