 7c3f847aad
			
		
	
	7c3f847aad
	
	
	
		
			
			Running "git check-ref-format --branch @{-1}" from outside any
repository produces
	$ git check-ref-format --branch @{-1}
	BUG: environment.c:182: git environment hasn't been setup
This is because the expansion of @{-1} must come from the HEAD reflog,
which involves opening the repository.  @{u} and @{push} (which are
more unusual because they typically would not expand to a local
branch) trigger the same assertion.
This has been broken since day one.  Before v2.13.0-rc0~48^2
(setup_git_env: avoid blind fall-back to ".git", 2016-10-02), the
breakage was more subtle: Git would read reflogs from ".git" within
the current directory even if it was not a valid repository.  Usually
that is harmless because Git is not being run from the root directory
of an invalid repository, but in edge cases such accesses can be
confusing or harmful.  Since v2.13.0, the problem is easier to
diagnose because Git aborts with a BUG message.
Erroring out is the right behavior: when asked to interpret a branch
name like "@{-1}", there is no reasonable answer in this context.
But we should print a message saying so instead of an assertion failure.
We do not forbid "check-ref-format --branch" from outside a repository
altogether because it is ok for a script to pre-process branch
arguments without @{...} in such a context.  For example, with
pre-2.13 Git, a script that does
	branch='master'; # default value
	parse_options
	branch=$(git check-ref-format --branch "$branch")
to normalize an optional branch name provided by the user would work
both inside a repository (where the user could provide '@{-1}') and
outside (where '@{-1}' should not be accepted).
So disable the "expand @{...}" half of the feature when run outside a
repository, but keep the check of the syntax of a proposed branch
name. This way, when run from outside a repository, "git
check-ref-format --branch @{-1}" will gracefully fail:
	$ git check-ref-format --branch @{-1}
	fatal: '@{-1}' is not a valid branch name
and "git check-ref-format --branch master" will succeed as before:
	$ git check-ref-format --branch master
	master
restoring the usual pre-2.13 behavior.
[jn: split out from a larger patch; moved conditional to
 strbuf_check_branch_ref instead of its caller; fleshed out commit
 message; some style tweaks in tests]
Reported-by: Marko Kungla <marko.kungla@gmail.com>
Helped-by: Jeff King <peff@peff.net>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
Signed-off-by: Jonathan Nieder <jrnieder@gmail.com>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
		
	
		
			
				
	
	
		
			219 lines
		
	
	
		
			5.8 KiB
		
	
	
	
		
			Bash
		
	
	
		
			Executable File
		
	
	
	
	
			
		
		
	
	
			219 lines
		
	
	
		
			5.8 KiB
		
	
	
	
		
			Bash
		
	
	
		
			Executable File
		
	
	
	
	
| #!/bin/sh
 | |
| 
 | |
| test_description='Test git check-ref-format'
 | |
| 
 | |
| . ./test-lib.sh
 | |
| 
 | |
| valid_ref() {
 | |
| 	prereq=
 | |
| 	case $1 in
 | |
| 	[A-Z!]*)
 | |
| 		prereq=$1
 | |
| 		shift
 | |
| 	esac
 | |
| 	desc="ref name '$1' is valid${2:+ with options $2}"
 | |
| 	test_expect_success $prereq "$desc" "
 | |
| 		git check-ref-format $2 '$1'
 | |
| 	"
 | |
| }
 | |
| invalid_ref() {
 | |
| 	prereq=
 | |
| 	case $1 in
 | |
| 	[A-Z!]*)
 | |
| 		prereq=$1
 | |
| 		shift
 | |
| 	esac
 | |
| 	desc="ref name '$1' is invalid${2:+ with options $2}"
 | |
| 	test_expect_success $prereq "$desc" "
 | |
| 		test_must_fail git check-ref-format $2 '$1'
 | |
| 	"
 | |
| }
 | |
| 
 | |
| invalid_ref ''
 | |
| invalid_ref !MINGW '/'
 | |
| invalid_ref !MINGW '/' --allow-onelevel
 | |
| invalid_ref !MINGW '/' --normalize
 | |
| invalid_ref !MINGW '/' '--allow-onelevel --normalize'
 | |
| valid_ref 'foo/bar/baz'
 | |
| valid_ref 'foo/bar/baz' --normalize
 | |
| invalid_ref 'refs///heads/foo'
 | |
| valid_ref 'refs///heads/foo' --normalize
 | |
| invalid_ref 'heads/foo/'
 | |
| invalid_ref !MINGW '/heads/foo'
 | |
| valid_ref !MINGW '/heads/foo' --normalize
 | |
| invalid_ref '///heads/foo'
 | |
| valid_ref '///heads/foo' --normalize
 | |
| invalid_ref './foo'
 | |
| invalid_ref './foo/bar'
 | |
| invalid_ref 'foo/./bar'
 | |
| invalid_ref 'foo/bar/.'
 | |
| invalid_ref '.refs/foo'
 | |
| invalid_ref 'refs/heads/foo.'
 | |
| invalid_ref 'heads/foo..bar'
 | |
| invalid_ref 'heads/foo?bar'
 | |
| valid_ref 'foo./bar'
 | |
| invalid_ref 'heads/foo.lock'
 | |
| invalid_ref 'heads///foo.lock'
 | |
| invalid_ref 'foo.lock/bar'
 | |
| invalid_ref 'foo.lock///bar'
 | |
| valid_ref 'heads/foo@bar'
 | |
| invalid_ref 'heads/v@{ation'
 | |
| invalid_ref 'heads/foo\bar'
 | |
| invalid_ref "$(printf 'heads/foo\t')"
 | |
| invalid_ref "$(printf 'heads/foo\177')"
 | |
| valid_ref "$(printf 'heads/fu\303\237')"
 | |
| valid_ref 'heads/*foo/bar' --refspec-pattern
 | |
| valid_ref 'heads/foo*/bar' --refspec-pattern
 | |
| valid_ref 'heads/f*o/bar' --refspec-pattern
 | |
| invalid_ref 'heads/f*o*/bar' --refspec-pattern
 | |
| invalid_ref 'heads/foo*/bar*' --refspec-pattern
 | |
| 
 | |
| ref='foo'
 | |
| invalid_ref "$ref"
 | |
| valid_ref "$ref" --allow-onelevel
 | |
| invalid_ref "$ref" --refspec-pattern
 | |
| valid_ref "$ref" '--refspec-pattern --allow-onelevel'
 | |
| invalid_ref "$ref" --normalize
 | |
| valid_ref "$ref" '--allow-onelevel --normalize'
 | |
| 
 | |
| ref='foo/bar'
 | |
| valid_ref "$ref"
 | |
| valid_ref "$ref" --allow-onelevel
 | |
| valid_ref "$ref" --refspec-pattern
 | |
| valid_ref "$ref" '--refspec-pattern --allow-onelevel'
 | |
| valid_ref "$ref" --normalize
 | |
| 
 | |
| ref='foo/*'
 | |
| invalid_ref "$ref"
 | |
| invalid_ref "$ref" --allow-onelevel
 | |
| valid_ref "$ref" --refspec-pattern
 | |
| valid_ref "$ref" '--refspec-pattern --allow-onelevel'
 | |
| 
 | |
| ref='*/foo'
 | |
| invalid_ref "$ref"
 | |
| invalid_ref "$ref" --allow-onelevel
 | |
| valid_ref "$ref" --refspec-pattern
 | |
| valid_ref "$ref" '--refspec-pattern --allow-onelevel'
 | |
| invalid_ref "$ref" --normalize
 | |
| valid_ref "$ref" '--refspec-pattern --normalize'
 | |
| 
 | |
| ref='foo/*/bar'
 | |
| invalid_ref "$ref"
 | |
| invalid_ref "$ref" --allow-onelevel
 | |
| valid_ref "$ref" --refspec-pattern
 | |
| valid_ref "$ref" '--refspec-pattern --allow-onelevel'
 | |
| 
 | |
| ref='*'
 | |
| invalid_ref "$ref"
 | |
| invalid_ref "$ref" --allow-onelevel
 | |
| invalid_ref "$ref" --refspec-pattern
 | |
| valid_ref "$ref" '--refspec-pattern --allow-onelevel'
 | |
| 
 | |
| ref='foo/*/*'
 | |
| invalid_ref "$ref" --refspec-pattern
 | |
| invalid_ref "$ref" '--refspec-pattern --allow-onelevel'
 | |
| 
 | |
| ref='*/foo/*'
 | |
| invalid_ref "$ref" --refspec-pattern
 | |
| invalid_ref "$ref" '--refspec-pattern --allow-onelevel'
 | |
| 
 | |
| ref='*/*/foo'
 | |
| invalid_ref "$ref" --refspec-pattern
 | |
| invalid_ref "$ref" '--refspec-pattern --allow-onelevel'
 | |
| 
 | |
| ref='/foo'
 | |
| invalid_ref !MINGW "$ref"
 | |
| invalid_ref !MINGW "$ref" --allow-onelevel
 | |
| invalid_ref !MINGW "$ref" --refspec-pattern
 | |
| invalid_ref !MINGW "$ref" '--refspec-pattern --allow-onelevel'
 | |
| invalid_ref !MINGW "$ref" --normalize
 | |
| valid_ref !MINGW "$ref" '--allow-onelevel --normalize'
 | |
| invalid_ref !MINGW "$ref" '--refspec-pattern --normalize'
 | |
| valid_ref !MINGW "$ref" '--refspec-pattern --allow-onelevel --normalize'
 | |
| 
 | |
| test_expect_success "check-ref-format --branch @{-1}" '
 | |
| 	T=$(git write-tree) &&
 | |
| 	sha1=$(echo A | git commit-tree $T) &&
 | |
| 	git update-ref refs/heads/master $sha1 &&
 | |
| 	git update-ref refs/remotes/origin/master $sha1 &&
 | |
| 	git checkout master &&
 | |
| 	git checkout origin/master &&
 | |
| 	git checkout master &&
 | |
| 	refname=$(git check-ref-format --branch @{-1}) &&
 | |
| 	test "$refname" = "$sha1" &&
 | |
| 	refname2=$(git check-ref-format --branch @{-2}) &&
 | |
| 	test "$refname2" = master'
 | |
| 
 | |
| test_expect_success 'check-ref-format --branch -naster' '
 | |
| 	test_must_fail git check-ref-format --branch -naster >actual &&
 | |
| 	test_must_be_empty actual
 | |
| '
 | |
| 
 | |
| test_expect_success 'check-ref-format --branch from subdir' '
 | |
| 	mkdir subdir &&
 | |
| 
 | |
| 	T=$(git write-tree) &&
 | |
| 	sha1=$(echo A | git commit-tree $T) &&
 | |
| 	git update-ref refs/heads/master $sha1 &&
 | |
| 	git update-ref refs/remotes/origin/master $sha1 &&
 | |
| 	git checkout master &&
 | |
| 	git checkout origin/master &&
 | |
| 	git checkout master &&
 | |
| 	refname=$(
 | |
| 		cd subdir &&
 | |
| 		git check-ref-format --branch @{-1}
 | |
| 	) &&
 | |
| 	test "$refname" = "$sha1"
 | |
| '
 | |
| 
 | |
| test_expect_success 'check-ref-format --branch @{-1} from non-repo' '
 | |
| 	nongit test_must_fail git check-ref-format --branch @{-1} >actual &&
 | |
| 	test_must_be_empty actual
 | |
| '
 | |
| 
 | |
| test_expect_success 'check-ref-format --branch master from non-repo' '
 | |
| 	echo master >expect &&
 | |
| 	nongit git check-ref-format --branch master >actual &&
 | |
| 	test_cmp expect actual
 | |
| '
 | |
| 
 | |
| valid_ref_normalized() {
 | |
| 	prereq=
 | |
| 	case $1 in
 | |
| 	[A-Z!]*)
 | |
| 		prereq=$1
 | |
| 		shift
 | |
| 	esac
 | |
| 	test_expect_success $prereq "ref name '$1' simplifies to '$2'" "
 | |
| 		refname=\$(git check-ref-format --normalize '$1') &&
 | |
| 		test \"\$refname\" = '$2'
 | |
| 	"
 | |
| }
 | |
| invalid_ref_normalized() {
 | |
| 	prereq=
 | |
| 	case $1 in
 | |
| 	[A-Z!]*)
 | |
| 		prereq=$1
 | |
| 		shift
 | |
| 	esac
 | |
| 	test_expect_success $prereq "check-ref-format --normalize rejects '$1'" "
 | |
| 		test_must_fail git check-ref-format --normalize '$1'
 | |
| 	"
 | |
| }
 | |
| 
 | |
| valid_ref_normalized 'heads/foo' 'heads/foo'
 | |
| valid_ref_normalized 'refs///heads/foo' 'refs/heads/foo'
 | |
| valid_ref_normalized !MINGW '/heads/foo' 'heads/foo'
 | |
| valid_ref_normalized '///heads/foo' 'heads/foo'
 | |
| invalid_ref_normalized 'foo'
 | |
| invalid_ref_normalized !MINGW '/foo'
 | |
| invalid_ref_normalized 'heads/foo/../bar'
 | |
| invalid_ref_normalized 'heads/./foo'
 | |
| invalid_ref_normalized 'heads\foo'
 | |
| invalid_ref_normalized 'heads/foo.lock'
 | |
| invalid_ref_normalized 'heads///foo.lock'
 | |
| invalid_ref_normalized 'foo.lock/bar'
 | |
| invalid_ref_normalized 'foo.lock///bar'
 | |
| 
 | |
| test_done
 |