commit: detect misspelled pathspec while making a partial commit.

When you say "git commit Documentaiton" to make partial commit
for the files only in that directory, we did not detect that as
a misspelled pathname and attempted to commit index without
change.  If nothing matched, there is no harm done, but if the
index gets modified otherwise by having another valid pathspec
or after an explicit update-index, a user will not notice
without paying attention to the "git status" preview.

This introduces --error-unmatch option to ls-files, and uses it
to detect this common user error.

Signed-off-by: Junio C Hamano <junkio@cox.net>
This commit is contained in:
Junio C Hamano 2006-02-14 12:40:20 -08:00
parent 9ece7169a4
commit bba319b5ce
2 changed files with 53 additions and 17 deletions

View File

@ -180,6 +180,7 @@ verify=t
verbose= verbose=
signoff= signoff=
force_author= force_author=
only_include_assumed=
while case "$#" in 0) break;; esac while case "$#" in 0) break;; esac
do do
case "$1" in case "$1" in
@ -340,15 +341,8 @@ case "$#,$also$only" in
0,) 0,)
;; ;;
*,) *,)
echo >&2 "assuming --include paths..." only_include_assumed="# Explicit paths specified without -i nor -o; assuming --include paths..."
also=t also=t
# Later when switch the defaults, we will replace them with these:
# echo >&2 "assuming --only paths..."
# also=
# If we are going to launch an editor, the message won't be
# shown without this...
test -z "$log_given$status_only" && sleep 1
;; ;;
esac esac
unset only unset only
@ -383,6 +377,8 @@ t,)
;; ;;
,t) ,t)
save_index && save_index &&
git-ls-files --error-unmatch -- "$@" >/dev/null || exit
git-diff-files --name-only -z -- "$@" | git-diff-files --name-only -z -- "$@" |
( (
cd "$TOP" cd "$TOP"
@ -411,7 +407,7 @@ t,)
refuse_partial "Different in index and the last commit: refuse_partial "Different in index and the last commit:
$dirty_in_index" $dirty_in_index"
fi fi
commit_only=`git-ls-files -- "$@"` commit_only=`git-ls-files --error-unmatch -- "$@"` || exit
# Build the temporary index and update the real index # Build the temporary index and update the real index
# the same way. # the same way.
@ -572,7 +568,10 @@ else
PARENTS="" PARENTS=""
fi fi
run_status >>"$GIT_DIR"/COMMIT_EDITMSG {
test -z "$only_include_assumed" || echo "$only_include_assumed"
run_status
} >>"$GIT_DIR"/COMMIT_EDITMSG
if [ "$?" != "0" -a ! -f "$GIT_DIR/MERGE_HEAD" ] if [ "$?" != "0" -a ! -f "$GIT_DIR/MERGE_HEAD" ]
then then
rm -f "$GIT_DIR/COMMIT_EDITMSG" rm -f "$GIT_DIR/COMMIT_EDITMSG"

View File

@ -25,6 +25,8 @@ static int line_terminator = '\n';
static int prefix_len = 0, prefix_offset = 0; static int prefix_len = 0, prefix_offset = 0;
static const char *prefix = NULL; static const char *prefix = NULL;
static const char **pathspec = NULL; static const char **pathspec = NULL;
static int error_unmatch = 0;
static char *ps_matched = NULL;
static const char *tag_cached = ""; static const char *tag_cached = "";
static const char *tag_unmerged = ""; static const char *tag_unmerged = "";
@ -325,7 +327,8 @@ static int cmp_name(const void *p1, const void *p2)
* Match a pathspec against a filename. The first "len" characters * Match a pathspec against a filename. The first "len" characters
* are the common prefix * are the common prefix
*/ */
static int match(const char **spec, const char *filename, int len) static int match(const char **spec, char *ps_matched,
const char *filename, int len)
{ {
const char *m; const char *m;
@ -333,16 +336,23 @@ static int match(const char **spec, const char *filename, int len)
int matchlen = strlen(m + len); int matchlen = strlen(m + len);
if (!matchlen) if (!matchlen)
return 1; goto matched;
if (!strncmp(m + len, filename + len, matchlen)) { if (!strncmp(m + len, filename + len, matchlen)) {
if (m[len + matchlen - 1] == '/') if (m[len + matchlen - 1] == '/')
return 1; goto matched;
switch (filename[len + matchlen]) { switch (filename[len + matchlen]) {
case '/': case '\0': case '/': case '\0':
return 1; goto matched;
} }
} }
if (!fnmatch(m + len, filename + len, 0)) if (!fnmatch(m + len, filename + len, 0))
goto matched;
if (ps_matched)
ps_matched++;
continue;
matched:
if (ps_matched)
*ps_matched = 1;
return 1; return 1;
} }
return 0; return 0;
@ -356,7 +366,7 @@ static void show_dir_entry(const char *tag, struct nond_on_fs *ent)
if (len >= ent->len) if (len >= ent->len)
die("git-ls-files: internal error - directory entry not superset of prefix"); die("git-ls-files: internal error - directory entry not superset of prefix");
if (pathspec && !match(pathspec, ent->name, len)) if (pathspec && !match(pathspec, ps_matched, ent->name, len))
return; return;
fputs(tag, stdout); fputs(tag, stdout);
@ -444,7 +454,7 @@ static void show_ce_entry(const char *tag, struct cache_entry *ce)
if (len >= ce_namelen(ce)) if (len >= ce_namelen(ce))
die("git-ls-files: internal error - cache entry not superset of prefix"); die("git-ls-files: internal error - cache entry not superset of prefix");
if (pathspec && !match(pathspec, ce->name, len)) if (pathspec && !match(pathspec, ps_matched, ce->name, len))
return; return;
if (!show_stage) { if (!show_stage) {
@ -699,6 +709,10 @@ int main(int argc, const char **argv)
prefix_offset = 0; prefix_offset = 0;
continue; continue;
} }
if (!strcmp(arg, "--error-unmatch")) {
error_unmatch = 1;
continue;
}
if (*arg == '-') if (*arg == '-')
usage(ls_files_usage); usage(ls_files_usage);
break; break;
@ -710,6 +724,14 @@ int main(int argc, const char **argv)
if (pathspec) if (pathspec)
verify_pathspec(); verify_pathspec();
/* Treat unmatching pathspec elements as errors */
if (pathspec && error_unmatch) {
int num;
for (num = 0; pathspec[num]; num++)
;
ps_matched = xcalloc(1, num);
}
if (show_ignored && !exc_given) { if (show_ignored && !exc_given) {
fprintf(stderr, "%s: --ignored needs some exclude pattern\n", fprintf(stderr, "%s: --ignored needs some exclude pattern\n",
argv[0]); argv[0]);
@ -725,5 +747,20 @@ int main(int argc, const char **argv)
if (prefix) if (prefix)
prune_cache(); prune_cache();
show_files(); show_files();
if (ps_matched) {
/* We need to make sure all pathspec matched otherwise
* it is an error.
*/
int num, errors = 0;
for (num = 0; pathspec[num]; num++) {
if (ps_matched[num])
continue;
error("pathspec '%s' did not match any.",
pathspec[num] + prefix_len);
}
return errors ? 1 : 0;
}
return 0; return 0;
} }