Merge branch 'pb/ref-filter-with-crlf'
A commit and tag object may have CR at the end of each and every line (you can create such an object with hash-object or using --cleanup=verbatim to decline the default clean-up action), but it would make it impossible to have a blank line to separate the title from the body of the message. Be lenient and accept a line with lone CR on it as a blank line, too. * pb/ref-filter-with-crlf: log, show: add tests for messages containing CRLF ref-filter: handle CRLF at end-of-line more gracefully
This commit is contained in:
		
							
								
								
									
										34
									
								
								ref-filter.c
									
									
									
									
									
								
							
							
						
						
									
										34
									
								
								ref-filter.c
									
									
									
									
									
								
							| @ -1097,14 +1097,19 @@ static const char *copy_email(const char *buf, struct used_atom *atom) | |||||||
|  |  | ||||||
| static char *copy_subject(const char *buf, unsigned long len) | static char *copy_subject(const char *buf, unsigned long len) | ||||||
| { | { | ||||||
| 	char *r = xmemdupz(buf, len); | 	struct strbuf sb = STRBUF_INIT; | ||||||
| 	int i; | 	int i; | ||||||
|  |  | ||||||
| 	for (i = 0; i < len; i++) | 	for (i = 0; i < len; i++) { | ||||||
| 		if (r[i] == '\n') | 		if (buf[i] == '\r' && i + 1 < len && buf[i + 1] == '\n') | ||||||
| 			r[i] = ' '; | 			continue; /* ignore CR in CRLF */ | ||||||
|  |  | ||||||
| 	return r; | 		if (buf[i] == '\n') | ||||||
|  | 			strbuf_addch(&sb, ' '); | ||||||
|  | 		else | ||||||
|  | 			strbuf_addch(&sb, buf[i]); | ||||||
|  | 	} | ||||||
|  | 	return strbuf_detach(&sb, NULL); | ||||||
| } | } | ||||||
|  |  | ||||||
| static void grab_date(const char *buf, struct atom_value *v, const char *atomname) | static void grab_date(const char *buf, struct atom_value *v, const char *atomname) | ||||||
| @ -1228,20 +1233,23 @@ static void find_subpos(const char *buf, | |||||||
|  |  | ||||||
| 	/* subject is first non-empty line */ | 	/* subject is first non-empty line */ | ||||||
| 	*sub = buf; | 	*sub = buf; | ||||||
| 	/* subject goes to first empty line */ | 	/* subject goes to first empty line before signature begins */ | ||||||
| 	while (buf < *sig && *buf && *buf != '\n') { | 	if ((eol = strstr(*sub, "\n\n"))) { | ||||||
| 		eol = strchrnul(buf, '\n'); | 		eol = eol < *sig ? eol : *sig; | ||||||
| 		if (*eol) | 	/* check if message uses CRLF */ | ||||||
| 			eol++; | 	} else if (! (eol = strstr(*sub, "\r\n\r\n"))) { | ||||||
| 		buf = eol; | 		/* treat whole message as subject */ | ||||||
|  | 		eol = strrchr(*sub, '\0'); | ||||||
| 	} | 	} | ||||||
|  | 	buf = eol; | ||||||
| 	*sublen = buf - *sub; | 	*sublen = buf - *sub; | ||||||
| 	/* drop trailing newline, if present */ | 	/* drop trailing newline, if present */ | ||||||
| 	if (*sublen && (*sub)[*sublen - 1] == '\n') | 	while (*sublen && ((*sub)[*sublen - 1] == '\n' || | ||||||
|  | 			   (*sub)[*sublen - 1] == '\r')) | ||||||
| 		*sublen -= 1; | 		*sublen -= 1; | ||||||
|  |  | ||||||
| 	/* skip any empty lines */ | 	/* skip any empty lines */ | ||||||
| 	while (*buf == '\n') | 	while (*buf == '\n' || *buf == '\r') | ||||||
| 		buf++; | 		buf++; | ||||||
| 	*body = buf; | 	*body = buf; | ||||||
| 	*bodylen = strlen(buf); | 	*bodylen = strlen(buf); | ||||||
|  | |||||||
							
								
								
									
										126
									
								
								t/t3920-crlf-messages.sh
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										126
									
								
								t/t3920-crlf-messages.sh
									
									
									
									
									
										Executable file
									
								
							| @ -0,0 +1,126 @@ | |||||||
|  | #!/bin/sh | ||||||
|  |  | ||||||
|  | test_description='Test ref-filter and pretty APIs for commit and tag messages using CRLF' | ||||||
|  | . ./test-lib.sh | ||||||
|  |  | ||||||
|  | LIB_CRLF_BRANCHES="" | ||||||
|  |  | ||||||
|  | create_crlf_ref () { | ||||||
|  | 	branch="$1" && | ||||||
|  | 	cat >.crlf-orig-$branch.txt && | ||||||
|  | 	cat .crlf-orig-$branch.txt | append_cr >.crlf-message-$branch.txt && | ||||||
|  | 	grep 'Subject' .crlf-orig-$branch.txt | tr '\n' ' ' | sed 's/[ ]*$//' | tr -d '\n' >.crlf-subject-$branch.txt && | ||||||
|  | 	grep 'Body' .crlf-message-$branch.txt >.crlf-body-$branch.txt || true && | ||||||
|  | 	LIB_CRLF_BRANCHES="${LIB_CRLF_BRANCHES} ${branch}" && | ||||||
|  | 	test_tick && | ||||||
|  | 	hash=$(git commit-tree HEAD^{tree} -p HEAD -F .crlf-message-${branch}.txt) && | ||||||
|  | 	git branch ${branch} ${hash} && | ||||||
|  | 	git tag tag-${branch} ${branch} -F .crlf-message-${branch}.txt --cleanup=verbatim | ||||||
|  | } | ||||||
|  |  | ||||||
|  | create_crlf_refs () { | ||||||
|  | 	create_crlf_ref crlf <<-\EOF && | ||||||
|  | 	Subject first line | ||||||
|  |  | ||||||
|  | 	Body first line | ||||||
|  | 	Body second line | ||||||
|  | 	EOF | ||||||
|  | 	create_crlf_ref crlf-empty-lines-after-subject <<-\EOF && | ||||||
|  | 	Subject first line | ||||||
|  |  | ||||||
|  |  | ||||||
|  | 	Body first line | ||||||
|  | 	Body second line | ||||||
|  | 	EOF | ||||||
|  | 	create_crlf_ref crlf-two-line-subject <<-\EOF && | ||||||
|  | 	Subject first line | ||||||
|  | 	Subject second line | ||||||
|  |  | ||||||
|  | 	Body first line | ||||||
|  | 	Body second line | ||||||
|  | 	EOF | ||||||
|  | 	create_crlf_ref crlf-two-line-subject-no-body <<-\EOF && | ||||||
|  | 	Subject first line | ||||||
|  | 	Subject second line | ||||||
|  | 	EOF | ||||||
|  | 	create_crlf_ref crlf-two-line-subject-no-body-trailing-newline <<-\EOF | ||||||
|  | 	Subject first line | ||||||
|  | 	Subject second line | ||||||
|  |  | ||||||
|  | 	EOF | ||||||
|  | } | ||||||
|  |  | ||||||
|  | test_crlf_subject_body_and_contents() { | ||||||
|  | 	command_and_args="$@" && | ||||||
|  | 	command=$1 && | ||||||
|  | 	if test ${command} = "branch" || test ${command} = "for-each-ref" || test ${command} = "tag" | ||||||
|  | 	then | ||||||
|  | 		atoms="(contents:subject) (contents:body) (contents)" | ||||||
|  | 	elif test ${command} = "log" || test ${command} = "show" | ||||||
|  | 	then | ||||||
|  | 		atoms="s b B" | ||||||
|  | 	fi && | ||||||
|  | 	files="subject body message" && | ||||||
|  | 	while test -n "${atoms}" | ||||||
|  | 	do | ||||||
|  | 		set ${atoms} && atom=$1 && shift && atoms="$*" && | ||||||
|  | 		set ${files} &&	file=$1 && shift && files="$*" && | ||||||
|  | 		test_expect_success "${command}: --format='%${atom}' works with messages using CRLF" " | ||||||
|  | 			rm -f expect && | ||||||
|  | 			for ref in ${LIB_CRLF_BRANCHES} | ||||||
|  | 			do | ||||||
|  | 				cat .crlf-${file}-\"\${ref}\".txt >>expect && | ||||||
|  | 				printf \"\n\" >>expect | ||||||
|  | 			done && | ||||||
|  | 			git $command_and_args --format=\"%${atom}\" >actual && | ||||||
|  | 			test_cmp expect actual | ||||||
|  | 		" | ||||||
|  | 	done | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | test_expect_success 'Setup refs with commit and tag messages using CRLF' ' | ||||||
|  | 	test_commit inital && | ||||||
|  | 	create_crlf_refs | ||||||
|  | ' | ||||||
|  |  | ||||||
|  | test_expect_success 'branch: --verbose works with messages using CRLF' ' | ||||||
|  | 	rm -f expect && | ||||||
|  | 	for branch in $LIB_CRLF_BRANCHES | ||||||
|  | 	do | ||||||
|  | 		printf "  " >>expect && | ||||||
|  | 		cat .crlf-subject-${branch}.txt >>expect && | ||||||
|  | 		printf "\n" >>expect | ||||||
|  | 	done && | ||||||
|  | 	git branch -v >tmp && | ||||||
|  | 	# Remove first two columns, and the line for the currently checked out branch | ||||||
|  | 	current=$(git branch --show-current) && | ||||||
|  | 	grep -v $current <tmp | awk "{\$1=\$2=\"\"}1"  >actual && | ||||||
|  | 	test_cmp expect actual | ||||||
|  | ' | ||||||
|  |  | ||||||
|  | test_crlf_subject_body_and_contents branch --list crlf* | ||||||
|  |  | ||||||
|  | test_crlf_subject_body_and_contents tag --list tag-crlf* | ||||||
|  |  | ||||||
|  | test_crlf_subject_body_and_contents for-each-ref refs/heads/crlf* | ||||||
|  |  | ||||||
|  | test_expect_success 'log: --oneline works with messages using CRLF' ' | ||||||
|  | 	for branch in $LIB_CRLF_BRANCHES | ||||||
|  | 	do | ||||||
|  | 		cat .crlf-subject-${branch}.txt >expect && | ||||||
|  | 		printf "\n" >>expect && | ||||||
|  | 		git log --oneline -1 ${branch} >tmp-branch && | ||||||
|  | 		git log --oneline -1 tag-${branch} >tmp-tag && | ||||||
|  | 		cut -d" " -f2- <tmp-branch >actual-branch && | ||||||
|  | 		cut -d" " -f2- <tmp-tag >actual-tag && | ||||||
|  | 		test_cmp expect actual-branch && | ||||||
|  | 		test_cmp expect actual-tag || return 1 | ||||||
|  | 	done | ||||||
|  | ' | ||||||
|  |  | ||||||
|  | test_crlf_subject_body_and_contents log --all --reverse --grep Subject | ||||||
|  |  | ||||||
|  | test_crlf_subject_body_and_contents show $LIB_CRLF_BRANCHES | ||||||
|  |  | ||||||
|  | test_done | ||||||
		Reference in New Issue
	
	Block a user
	 Junio C Hamano
					Junio C Hamano