diff options
| author | Linus Torvalds <torvalds@osdl.org> | 2006-05-18 12:07:31 -0700 | 
|---|---|---|
| committer | Junio C Hamano <junkio@cox.net> | 2006-05-18 12:07:31 -0700 | 
| commit | 8dcf39c46e2931ca02b18b1ea3a6b21f446d8de8 (patch) | |
| tree | faa37fc9e41acdfb704ece258315fd37ddc6bcb4 /read-cache.c | |
| parent | e8f990b4e4b56f214138cc475c19e5a253e9148e (diff) | |
| download | git-8dcf39c46e2931ca02b18b1ea3a6b21f446d8de8.tar.gz | |
Prevent bogus paths from being added to the index.
With this one, it's now a fatal error to try to add a pathname
that cannot be added with "git add", i.e.
	[torvalds@g5 git]$ git add .git/config
	fatal: unable to add .git/config to index
and
	[torvalds@g5 git]$ git add foo/../bar
	fatal: unable to add foo/../bar to index
instead of the old "Ignoring path xyz" warning that would end up
silently succeeding on any other paths.
Signed-off-by: Linus Torvalds <torvalds@osdl.org>
Signed-off-by: Junio C Hamano <junkio@cox.net>
Diffstat (limited to 'read-cache.c')
| -rw-r--r-- | read-cache.c | 66 | 
1 files changed, 66 insertions, 0 deletions
| diff --git a/read-cache.c b/read-cache.c index a917ab0cfe..e8fa6d0812 100644 --- a/read-cache.c +++ b/read-cache.c @@ -332,6 +332,70 @@ int ce_path_match(const struct cache_entry *ce, const char **pathspec)  }  /* + * We fundamentally don't like some paths: we don't want + * dot or dot-dot anywhere, and for obvious reasons don't + * want to recurse into ".git" either. + * + * Also, we don't want double slashes or slashes at the + * end that can make pathnames ambiguous. + */ +static int verify_dotfile(const char *rest) +{ +	/* +	 * The first character was '.', but that +	 * has already been discarded, we now test +	 * the rest. +	 */ +	switch (*rest) { +	/* "." is not allowed */ +	case '\0': case '/': +		return 0; + +	/* +	 * ".git" followed by  NUL or slash is bad. This +	 * shares the path end test with the ".." case. +	 */ +	case 'g': +		if (rest[1] != 'i') +			break; +		if (rest[2] != 't') +			break; +		rest += 2; +	/* fallthrough */ +	case '.': +		if (rest[1] == '\0' || rest[1] == '/') +			return 0; +	} +	return 1; +} + +int verify_path(const char *path) +{ +	char c; + +	goto inside; +	for (;;) { +		if (!c) +			return 1; +		if (c == '/') { +inside: +			c = *path++; +			switch (c) { +			default: +				continue; +			case '/': case '\0': +				break; +			case '.': +				if (verify_dotfile(path)) +					continue; +			} +			return 0; +		} +		c = *path++; +	} +} + +/*   * Do we have another file that has the beginning components being a   * proper superset of the name we're trying to add?   */ @@ -472,6 +536,8 @@ int add_cache_entry(struct cache_entry *ce, int option)  	if (!ok_to_add)  		return -1; +	if (!verify_path(ce->name)) +		return -1;  	if (!skip_df_check &&  	    check_file_directory_conflict(ce, pos, ok_to_replace)) { | 
