diff options
| -rw-r--r-- | builtin-ls-files.c | 11 | ||||
| -rw-r--r-- | builtin-mv.c | 4 | ||||
| -rw-r--r-- | setup.c | 158 | ||||
| -rwxr-xr-x | t/t7010-setup.sh | 117 | 
4 files changed, 245 insertions, 45 deletions
| diff --git a/builtin-ls-files.c b/builtin-ls-files.c index 0f0ab2da16..3801cf4cec 100644 --- a/builtin-ls-files.c +++ b/builtin-ls-files.c @@ -572,8 +572,17 @@ int cmd_ls_files(int argc, const char **argv, const char *prefix)  	pathspec = get_pathspec(prefix, argv + i);  	/* Verify that the pathspec matches the prefix */ -	if (pathspec) +	if (pathspec) { +		if (argc != i) { +			int cnt; +			for (cnt = 0; pathspec[cnt]; cnt++) +				; +			if (cnt != (argc - i)) +				exit(1); /* error message already given */ +		}  		prefix = verify_pathspec(prefix); +	} else if (argc != i) +		exit(1); /* error message already given */  	/* Treat unmatching pathspec elements as errors */  	if (pathspec && error_unmatch) { diff --git a/builtin-mv.c b/builtin-mv.c index 990e21355d..94f6dd2aad 100644 --- a/builtin-mv.c +++ b/builtin-mv.c @@ -164,7 +164,7 @@ int cmd_mv(int argc, const char **argv, const char *prefix)  				}  				dst = add_slash(dst); -				dst_len = strlen(dst) - 1; +				dst_len = strlen(dst);  				for (j = 0; j < last - first; j++) {  					const char *path = @@ -172,7 +172,7 @@ int cmd_mv(int argc, const char **argv, const char *prefix)  					source[argc + j] = path;  					destination[argc + j] =  						prefix_path(dst, dst_len, -							path + length); +							path + length + 1);  					modes[argc + j] = INDEX;  				}  				argc += last - first; @@ -4,51 +4,118 @@  static int inside_git_dir = -1;  static int inside_work_tree = -1; -const char *prefix_path(const char *prefix, int len, const char *path) +static int sanitary_path_copy(char *dst, const char *src)  { -	const char *orig = path; +	char *dst0 = dst; + +	if (*src == '/') { +		*dst++ = '/'; +		while (*src == '/') +			src++; +	} +  	for (;;) { -		char c; -		if (*path != '.') -			break; -		c = path[1]; -		/* "." */ -		if (!c) { -			path++; -			break; +		char c = *src; + +		/* +		 * A path component that begins with . could be +		 * special: +		 * (1) "." and ends   -- ignore and terminate. +		 * (2) "./"           -- ignore them, eat slash and continue. +		 * (3) ".." and ends  -- strip one and terminate. +		 * (4) "../"          -- strip one, eat slash and continue. +		 */ +		if (c == '.') { +			switch (src[1]) { +			case '\0': +				/* (1) */ +				src++; +				break; +			case '/': +				/* (2) */ +				src += 2; +				while (*src == '/') +					src++; +				continue; +			case '.': +				switch (src[2]) { +				case '\0': +					/* (3) */ +					src += 2; +					goto up_one; +				case '/': +					/* (4) */ +					src += 3; +					while (*src == '/') +						src++; +					goto up_one; +				} +			}  		} -		/* "./" */ + +		/* copy up to the next '/', and eat all '/' */ +		while ((c = *src++) != '\0' && c != '/') +			*dst++ = c;  		if (c == '/') { -			path += 2; -			continue; -		} -		if (c != '.') +			*dst++ = c; +			while (c == '/') +				c = *src++; +			src--; +		} else if (!c)  			break; -		c = path[2]; -		if (!c) -			path += 2; -		else if (c == '/') -			path += 3; -		else -			break; -		/* ".." and "../" */ -		/* Remove last component of the prefix */ -		do { -			if (!len) -				die("'%s' is outside repository", orig); -			len--; -		} while (len && prefix[len-1] != '/');  		continue; + +	up_one: +		/* +		 * dst0..dst is prefix portion, and dst[-1] is '/'; +		 * go up one level. +		 */ +		dst -= 2; /* go past trailing '/' if any */ +		if (dst < dst0) +			return -1; +		while (1) { +			if (dst <= dst0) +				break; +			c = *dst--; +			if (c == '/') { +				dst += 2; +				break; +			} +		}  	} -	if (len) { -		int speclen = strlen(path); -		char *n = xmalloc(speclen + len + 1); +	*dst = '\0'; +	return 0; +} -		memcpy(n, prefix, len); -		memcpy(n + len, path, speclen+1); -		path = n; +const char *prefix_path(const char *prefix, int len, const char *path) +{ +	const char *orig = path; +	char *sanitized = xmalloc(len + strlen(path) + 1); +	if (*orig == '/') +		strcpy(sanitized, path); +	else { +		if (len) +			memcpy(sanitized, prefix, len); +		strcpy(sanitized + len, path);  	} -	return path; +	if (sanitary_path_copy(sanitized, sanitized)) +		goto error_out; +	if (*orig == '/') { +		const char *work_tree = get_git_work_tree(); +		size_t len = strlen(work_tree); +		size_t total = strlen(sanitized) + 1; +		if (strncmp(sanitized, work_tree, len) || +		    (sanitized[len] != '\0' && sanitized[len] != '/')) { +		error_out: +			error("'%s' is outside repository", orig); +			free(sanitized); +			return NULL; +		} +		if (sanitized[len] == '/') +			len++; +		memmove(sanitized, sanitized + len, total - len); +	} +	return sanitized;  }  /* @@ -114,7 +181,7 @@ void verify_non_filename(const char *prefix, const char *arg)  const char **get_pathspec(const char *prefix, const char **pathspec)  {  	const char *entry = *pathspec; -	const char **p; +	const char **src, **dst;  	int prefixlen;  	if (!prefix && !entry) @@ -128,12 +195,19 @@ const char **get_pathspec(const char *prefix, const char **pathspec)  	}  	/* Otherwise we have to re-write the entries.. */ -	p = pathspec; +	src = pathspec; +	dst = pathspec;  	prefixlen = prefix ? strlen(prefix) : 0; -	do { -		*p = prefix_path(prefix, prefixlen, entry); -	} while ((entry = *++p) != NULL); -	return (const char **) pathspec; +	while (*src) { +		const char *p = prefix_path(prefix, prefixlen, *src); +		if (p) +			*(dst++) = p; +		src++; +	} +	*dst = NULL; +	if (!*pathspec) +		return NULL; +	return pathspec;  }  /* diff --git a/t/t7010-setup.sh b/t/t7010-setup.sh new file mode 100755 index 0000000000..da20ba514a --- /dev/null +++ b/t/t7010-setup.sh @@ -0,0 +1,117 @@ +#!/bin/sh + +test_description='setup taking and sanitizing funny paths' + +. ./test-lib.sh + +test_expect_success setup ' + +	mkdir -p a/b/c a/e && +	D=$(pwd) && +	>a/b/c/d && +	>a/e/f + +' + +test_expect_success 'git add (absolute)' ' + +	git add "$D/a/b/c/d" && +	git ls-files >current && +	echo a/b/c/d >expect && +	diff -u expect current + +' + + +test_expect_success 'git add (funny relative)' ' + +	rm -f .git/index && +	( +		cd a/b && +		git add "../e/./f" +	) && +	git ls-files >current && +	echo a/e/f >expect && +	diff -u expect current + +' + +test_expect_success 'git rm (absolute)' ' + +	rm -f .git/index && +	git add a && +	git rm -f --cached "$D/a/b/c/d" && +	git ls-files >current && +	echo a/e/f >expect && +	diff -u expect current + +' + +test_expect_success 'git rm (funny relative)' ' + +	rm -f .git/index && +	git add a && +	( +		cd a/b && +		git rm -f --cached "../e/./f" +	) && +	git ls-files >current && +	echo a/b/c/d >expect && +	diff -u expect current + +' + +test_expect_success 'git ls-files (absolute)' ' + +	rm -f .git/index && +	git add a && +	git ls-files "$D/a/e/../b" >current && +	echo a/b/c/d >expect && +	diff -u expect current + +' + +test_expect_success 'git ls-files (relative #1)' ' + +	rm -f .git/index && +	git add a && +	( +		cd a/b && +		git ls-files "../b/c" +	)  >current && +	echo c/d >expect && +	diff -u expect current + +' + +test_expect_success 'git ls-files (relative #2)' ' + +	rm -f .git/index && +	git add a && +	( +		cd a/b && +		git ls-files --full-name "../e/f" +	)  >current && +	echo a/e/f >expect && +	diff -u expect current + +' + +test_expect_success 'git ls-files (relative #3)' ' + +	rm -f .git/index && +	git add a && +	( +		cd a/b && +		if git ls-files "../e/f" +		then +			echo Gaah, should have failed +			exit 1 +		else +			: happy +		fi +	) + +' + +test_done | 
