| #!/bin/sh | 
 | # | 
 | # Copyright (c) 2009 Johan Herland | 
 | # | 
 |  | 
 | test_description='test git fast-import of notes objects' | 
 | . ./test-lib.sh | 
 |  | 
 |  | 
 | test_tick | 
 | cat >input <<INPUT_END | 
 | commit refs/heads/master | 
 | committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE | 
 | data <<COMMIT | 
 | first commit | 
 | COMMIT | 
 |  | 
 | M 644 inline foo | 
 | data <<EOF | 
 | file foo in first commit | 
 | EOF | 
 |  | 
 | M 755 inline bar | 
 | data <<EOF | 
 | file bar in first commit | 
 | EOF | 
 |  | 
 | M 644 inline baz/xyzzy | 
 | data <<EOF | 
 | file baz/xyzzy in first commit | 
 | EOF | 
 |  | 
 | commit refs/heads/master | 
 | committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE | 
 | data <<COMMIT | 
 | second commit | 
 | COMMIT | 
 |  | 
 | M 644 inline foo | 
 | data <<EOF | 
 | file foo in second commit | 
 | EOF | 
 |  | 
 | M 755 inline baz/xyzzy | 
 | data <<EOF | 
 | file baz/xyzzy in second commit | 
 | EOF | 
 |  | 
 | commit refs/heads/master | 
 | committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE | 
 | data <<COMMIT | 
 | third commit | 
 | COMMIT | 
 |  | 
 | M 644 inline foo | 
 | data <<EOF | 
 | file foo in third commit | 
 | EOF | 
 |  | 
 | commit refs/heads/master | 
 | committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE | 
 | data <<COMMIT | 
 | fourth commit | 
 | COMMIT | 
 |  | 
 | M 755 inline bar | 
 | data <<EOF | 
 | file bar in fourth commit | 
 | EOF | 
 |  | 
 | INPUT_END | 
 |  | 
 | test_expect_success 'set up master branch' ' | 
 |  | 
 | 	git fast-import <input && | 
 | 	git whatchanged master | 
 | ' | 
 |  | 
 | commit4=$(git rev-parse refs/heads/master) | 
 | commit3=$(git rev-parse "$commit4^") | 
 | commit2=$(git rev-parse "$commit4~2") | 
 | commit1=$(git rev-parse "$commit4~3") | 
 |  | 
 | test_tick | 
 | cat >input <<INPUT_END | 
 | commit refs/notes/test | 
 | committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE | 
 | data <<COMMIT | 
 | first notes commit | 
 | COMMIT | 
 |  | 
 | M 644 inline $commit1 | 
 | data <<EOF | 
 | first note for first commit | 
 | EOF | 
 |  | 
 | M 755 inline $commit2 | 
 | data <<EOF | 
 | first note for second commit | 
 | EOF | 
 |  | 
 | INPUT_END | 
 |  | 
 | cat >expect <<EXPECT_END | 
 |     fourth commit | 
 |     third commit | 
 |     second commit | 
 |     first note for second commit | 
 |     first commit | 
 |     first note for first commit | 
 | EXPECT_END | 
 |  | 
 | test_expect_success 'add notes with simple M command' ' | 
 |  | 
 | 	git fast-import <input && | 
 | 	GIT_NOTES_REF=refs/notes/test git log | grep "^    " > actual && | 
 | 	test_cmp expect actual | 
 |  | 
 | ' | 
 |  | 
 | test_tick | 
 | cat >input <<INPUT_END | 
 | feature notes | 
 | commit refs/notes/test | 
 | committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE | 
 | data <<COMMIT | 
 | second notes commit | 
 | COMMIT | 
 |  | 
 | from refs/notes/test^0 | 
 | N inline $commit3 | 
 | data <<EOF | 
 | first note for third commit | 
 | EOF | 
 |  | 
 | N inline $commit4 | 
 | data <<EOF | 
 | first note for fourth commit | 
 | EOF | 
 |  | 
 | INPUT_END | 
 |  | 
 | cat >expect <<EXPECT_END | 
 |     fourth commit | 
 |     first note for fourth commit | 
 |     third commit | 
 |     first note for third commit | 
 |     second commit | 
 |     first note for second commit | 
 |     first commit | 
 |     first note for first commit | 
 | EXPECT_END | 
 |  | 
 | test_expect_success 'add notes with simple N command' ' | 
 |  | 
 | 	git fast-import <input && | 
 | 	GIT_NOTES_REF=refs/notes/test git log | grep "^    " > actual && | 
 | 	test_cmp expect actual | 
 |  | 
 | ' | 
 |  | 
 | test_tick | 
 | cat >input <<INPUT_END | 
 | commit refs/notes/test | 
 | committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE | 
 | data <<COMMIT | 
 | third notes commit | 
 | COMMIT | 
 |  | 
 | from refs/notes/test^0 | 
 | N inline $commit1 | 
 | data <<EOF | 
 | second note for first commit | 
 | EOF | 
 |  | 
 | N inline $commit2 | 
 | data <<EOF | 
 | second note for second commit | 
 | EOF | 
 |  | 
 | N inline $commit3 | 
 | data <<EOF | 
 | second note for third commit | 
 | EOF | 
 |  | 
 | N inline $commit4 | 
 | data <<EOF | 
 | second note for fourth commit | 
 | EOF | 
 |  | 
 | INPUT_END | 
 |  | 
 | cat >expect <<EXPECT_END | 
 |     fourth commit | 
 |     second note for fourth commit | 
 |     third commit | 
 |     second note for third commit | 
 |     second commit | 
 |     second note for second commit | 
 |     first commit | 
 |     second note for first commit | 
 | EXPECT_END | 
 |  | 
 | test_expect_success 'update existing notes with N command' ' | 
 |  | 
 | 	git fast-import <input && | 
 | 	GIT_NOTES_REF=refs/notes/test git log | grep "^    " > actual && | 
 | 	test_cmp expect actual | 
 |  | 
 | ' | 
 |  | 
 | test_tick | 
 | cat >input <<INPUT_END | 
 | commit refs/notes/test | 
 | committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE | 
 | data <<COMMIT | 
 | fourth notes commit | 
 | COMMIT | 
 |  | 
 | from refs/notes/test^0 | 
 | M 644 inline $(echo "$commit3" | sed "s|^..|&/|") | 
 | data <<EOF | 
 | prefix of note for third commit | 
 | EOF | 
 |  | 
 | M 644 inline $(echo "$commit4" | sed "s|^..|&/|") | 
 | data <<EOF | 
 | prefix of note for fourth commit | 
 | EOF | 
 |  | 
 | M 644 inline $(echo "$commit4" | sed "s|^\(..\)\(..\)|\1/\2/|") | 
 | data <<EOF | 
 | pre-prefix of note for fourth commit | 
 | EOF | 
 |  | 
 | N inline $commit1 | 
 | data <<EOF | 
 | third note for first commit | 
 | EOF | 
 |  | 
 | N inline $commit2 | 
 | data <<EOF | 
 | third note for second commit | 
 | EOF | 
 |  | 
 | N inline $commit3 | 
 | data <<EOF | 
 | third note for third commit | 
 | EOF | 
 |  | 
 | N inline $commit4 | 
 | data <<EOF | 
 | third note for fourth commit | 
 | EOF | 
 |  | 
 |  | 
 | INPUT_END | 
 |  | 
 | whitespace="    " | 
 |  | 
 | cat >expect <<EXPECT_END | 
 |     fourth commit | 
 |     pre-prefix of note for fourth commit | 
 | $whitespace | 
 |     prefix of note for fourth commit | 
 | $whitespace | 
 |     third note for fourth commit | 
 |     third commit | 
 |     prefix of note for third commit | 
 | $whitespace | 
 |     third note for third commit | 
 |     second commit | 
 |     third note for second commit | 
 |     first commit | 
 |     third note for first commit | 
 | EXPECT_END | 
 |  | 
 | test_expect_success 'add concatentation notes with M command' ' | 
 |  | 
 | 	git fast-import <input && | 
 | 	GIT_NOTES_REF=refs/notes/test git log | grep "^    " > actual && | 
 | 	test_cmp expect actual | 
 |  | 
 | ' | 
 |  | 
 | test_tick | 
 | cat >input <<INPUT_END | 
 | commit refs/notes/test | 
 | committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE | 
 | data <<COMMIT | 
 | fifth notes commit | 
 | COMMIT | 
 |  | 
 | from refs/notes/test^0 | 
 | deleteall | 
 |  | 
 | INPUT_END | 
 |  | 
 | cat >expect <<EXPECT_END | 
 |     fourth commit | 
 |     third commit | 
 |     second commit | 
 |     first commit | 
 | EXPECT_END | 
 |  | 
 | test_expect_success 'verify that deleteall also removes notes' ' | 
 |  | 
 | 	git fast-import <input && | 
 | 	GIT_NOTES_REF=refs/notes/test git log | grep "^    " > actual && | 
 | 	test_cmp expect actual | 
 |  | 
 | ' | 
 |  | 
 | test_tick | 
 | cat >input <<INPUT_END | 
 | commit refs/notes/test | 
 | committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE | 
 | data <<COMMIT | 
 | sixth notes commit | 
 | COMMIT | 
 |  | 
 | from refs/notes/test^0 | 
 | M 644 inline $commit1 | 
 | data <<EOF | 
 | third note for first commit | 
 | EOF | 
 |  | 
 | M 644 inline $commit3 | 
 | data <<EOF | 
 | third note for third commit | 
 | EOF | 
 |  | 
 | N inline $commit1 | 
 | data <<EOF | 
 | fourth note for first commit | 
 | EOF | 
 |  | 
 | N inline $commit3 | 
 | data <<EOF | 
 | fourth note for third commit | 
 | EOF | 
 |  | 
 | INPUT_END | 
 |  | 
 | cat >expect <<EXPECT_END | 
 |     fourth commit | 
 |     third commit | 
 |     fourth note for third commit | 
 |     second commit | 
 |     first commit | 
 |     fourth note for first commit | 
 | EXPECT_END | 
 |  | 
 | test_expect_success 'verify that later N commands override earlier M commands' ' | 
 |  | 
 | 	git fast-import <input && | 
 | 	GIT_NOTES_REF=refs/notes/test git log | grep "^    " > actual && | 
 | 	test_cmp expect actual | 
 |  | 
 | ' | 
 |  | 
 | # Write fast-import commands to create the given number of commits | 
 | fast_import_commits () { | 
 | 	my_ref=$1 | 
 | 	my_num_commits=$2 | 
 | 	my_append_to_file=$3 | 
 | 	my_i=0 | 
 | 	while test $my_i -lt $my_num_commits | 
 | 	do | 
 | 		my_i=$(($my_i + 1)) | 
 | 		test_tick | 
 | 		cat >>"$my_append_to_file" <<INPUT_END | 
 | commit $my_ref | 
 | mark :$my_i | 
 | committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE | 
 | data <<COMMIT | 
 | commit #$my_i | 
 | COMMIT | 
 |  | 
 | M 644 inline file | 
 | data <<EOF | 
 | file contents in commit #$my_i | 
 | EOF | 
 |  | 
 | INPUT_END | 
 | 	done | 
 | } | 
 |  | 
 | # Write fast-import commands to create the given number of notes annotating | 
 | # the commits created by fast_import_commits() | 
 | fast_import_notes () { | 
 | 	my_notes_ref=$1 | 
 | 	my_num_commits=$2 | 
 | 	my_append_to_file=$3 | 
 | 	my_note_append=$4 | 
 | 	test_tick | 
 | 	cat >>"$my_append_to_file" <<INPUT_END | 
 | commit $my_notes_ref | 
 | committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE | 
 | data <<COMMIT | 
 | committing $my_num_commits notes | 
 | COMMIT | 
 |  | 
 | INPUT_END | 
 |  | 
 | 	my_i=0 | 
 | 	while test $my_i -lt $my_num_commits | 
 | 	do | 
 | 		my_i=$(($my_i + 1)) | 
 | 		cat >>"$my_append_to_file" <<INPUT_END | 
 | N inline :$my_i | 
 | data <<EOF | 
 | note for commit #$my_i$my_note_append | 
 | EOF | 
 |  | 
 | INPUT_END | 
 | 	done | 
 | } | 
 |  | 
 |  | 
 | rm input expect | 
 | num_commits=400 | 
 | # Create lots of commits | 
 | fast_import_commits "refs/heads/many_commits" $num_commits input | 
 | # Create one note per above commit | 
 | fast_import_notes "refs/notes/many_notes" $num_commits input | 
 | # Add a couple of non-notes as well | 
 | test_tick | 
 | cat >>input <<INPUT_END | 
 | commit refs/notes/many_notes | 
 | committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE | 
 | data <<COMMIT | 
 | committing some non-notes to the notes tree | 
 | COMMIT | 
 |  | 
 | M 755 inline foobar/non-note.txt | 
 | data <<EOF | 
 | This is not a note, but rather a regular file residing in a notes tree | 
 | EOF | 
 |  | 
 | M 644 inline deadbeef | 
 | data <<EOF | 
 | Non-note file | 
 | EOF | 
 |  | 
 | M 644 inline de/adbeef | 
 | data <<EOF | 
 | Another non-note file | 
 | EOF | 
 |  | 
 | INPUT_END | 
 | # Finally create the expected output from all these notes and commits | 
 | i=$num_commits | 
 | while test $i -gt 0 | 
 | do | 
 | 	cat >>expect <<EXPECT_END | 
 |     commit #$i | 
 |     note for commit #$i | 
 | EXPECT_END | 
 | 	i=$(($i - 1)) | 
 | done | 
 |  | 
 | test_expect_success 'add lots of commits and notes' ' | 
 |  | 
 | 	git fast-import <input && | 
 | 	GIT_NOTES_REF=refs/notes/many_notes git log refs/heads/many_commits | | 
 | 	    grep "^    " > actual && | 
 | 	test_cmp expect actual | 
 |  | 
 | ' | 
 |  | 
 | test_expect_success 'verify that lots of notes trigger a fanout scheme' ' | 
 |  | 
 | 	# None of the entries in the top-level notes tree should be a full SHA1 | 
 | 	git ls-tree --name-only refs/notes/many_notes | | 
 | 	while read path | 
 | 	do | 
 | 		if test $(expr length "$path") -ge 40 | 
 | 		then | 
 | 			return 1 | 
 | 		fi | 
 | 	done | 
 |  | 
 | ' | 
 |  | 
 | cat >>expect_non-note1 << EOF | 
 | This is not a note, but rather a regular file residing in a notes tree | 
 | EOF | 
 |  | 
 | cat >>expect_non-note2 << EOF | 
 | Non-note file | 
 | EOF | 
 |  | 
 | cat >>expect_non-note3 << EOF | 
 | Another non-note file | 
 | EOF | 
 |  | 
 | test_expect_success 'verify that non-notes are untouched by a fanout change' ' | 
 |  | 
 | 	git cat-file -p refs/notes/many_notes:foobar/non-note.txt > actual && | 
 | 	test_cmp expect_non-note1 actual && | 
 | 	git cat-file -p refs/notes/many_notes:deadbeef > actual && | 
 | 	test_cmp expect_non-note2 actual && | 
 | 	git cat-file -p refs/notes/many_notes:de/adbeef > actual && | 
 | 	test_cmp expect_non-note3 actual | 
 |  | 
 | ' | 
 |  | 
 | # Change the notes for the three top commits | 
 | test_tick | 
 | cat >input <<INPUT_END | 
 | commit refs/notes/many_notes | 
 | committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE | 
 | data <<COMMIT | 
 | changing notes for the top three commits | 
 | COMMIT | 
 | from refs/notes/many_notes^0 | 
 | INPUT_END | 
 |  | 
 | rm expect | 
 | i=$num_commits | 
 | j=0 | 
 | while test $j -lt 3 | 
 | do | 
 | 	cat >>input <<INPUT_END | 
 | N inline refs/heads/many_commits~$j | 
 | data <<EOF | 
 | changed note for commit #$i | 
 | EOF | 
 | INPUT_END | 
 | 	cat >>expect <<EXPECT_END | 
 |     commit #$i | 
 |     changed note for commit #$i | 
 | EXPECT_END | 
 | 	i=$(($i - 1)) | 
 | 	j=$(($j + 1)) | 
 | done | 
 |  | 
 | test_expect_success 'change a few existing notes' ' | 
 |  | 
 | 	git fast-import <input && | 
 | 	GIT_NOTES_REF=refs/notes/many_notes git log -n3 refs/heads/many_commits | | 
 | 	    grep "^    " > actual && | 
 | 	test_cmp expect actual | 
 |  | 
 | ' | 
 |  | 
 | test_expect_success 'verify that changing notes respect existing fanout' ' | 
 |  | 
 | 	# None of the entries in the top-level notes tree should be a full SHA1 | 
 | 	git ls-tree --name-only refs/notes/many_notes | | 
 | 	while read path | 
 | 	do | 
 | 		if test $(expr length "$path") -ge 40 | 
 | 		then | 
 | 			return 1 | 
 | 		fi | 
 | 	done | 
 |  | 
 | ' | 
 |  | 
 | remaining_notes=10 | 
 | test_tick | 
 | cat >input <<INPUT_END | 
 | commit refs/notes/many_notes | 
 | committer $GIT_COMMITTER_NAME <$GIT_COMMITTER_EMAIL> $GIT_COMMITTER_DATE | 
 | data <<COMMIT | 
 | removing all notes but $remaining_notes | 
 | COMMIT | 
 | from refs/notes/many_notes^0 | 
 | INPUT_END | 
 |  | 
 | i=$(($num_commits - $remaining_notes)) | 
 | for sha1 in $(git rev-list -n $i refs/heads/many_commits) | 
 | do | 
 | 	cat >>input <<INPUT_END | 
 | N 0000000000000000000000000000000000000000 $sha1 | 
 | INPUT_END | 
 | done | 
 |  | 
 | i=$num_commits | 
 | rm expect | 
 | while test $i -gt 0 | 
 | do | 
 | 	cat >>expect <<EXPECT_END | 
 |     commit #$i | 
 | EXPECT_END | 
 | 	if test $i -le $remaining_notes | 
 | 	then | 
 | 		cat >>expect <<EXPECT_END | 
 |     note for commit #$i | 
 | EXPECT_END | 
 | 	fi | 
 | 	i=$(($i - 1)) | 
 | done | 
 |  | 
 | test_expect_success 'remove lots of notes' ' | 
 |  | 
 | 	git fast-import <input && | 
 | 	GIT_NOTES_REF=refs/notes/many_notes git log refs/heads/many_commits | | 
 | 	    grep "^    " > actual && | 
 | 	test_cmp expect actual | 
 |  | 
 | ' | 
 |  | 
 | test_expect_success 'verify that removing notes trigger fanout consolidation' ' | 
 |  | 
 | 	# All entries in the top-level notes tree should be a full SHA1 | 
 | 	git ls-tree --name-only -r refs/notes/many_notes | | 
 | 	while read path | 
 | 	do | 
 | 		# Explicitly ignore the non-note paths | 
 | 		test "$path" = "foobar/non-note.txt" && continue | 
 | 		test "$path" = "deadbeef" && continue | 
 | 		test "$path" = "de/adbeef" && continue | 
 |  | 
 | 		if test $(expr length "$path") -ne 40 | 
 | 		then | 
 | 			return 1 | 
 | 		fi | 
 | 	done | 
 |  | 
 | ' | 
 |  | 
 | test_expect_success 'verify that non-notes are untouched by a fanout change' ' | 
 |  | 
 | 	git cat-file -p refs/notes/many_notes:foobar/non-note.txt > actual && | 
 | 	test_cmp expect_non-note1 actual && | 
 | 	git cat-file -p refs/notes/many_notes:deadbeef > actual && | 
 | 	test_cmp expect_non-note2 actual && | 
 | 	git cat-file -p refs/notes/many_notes:de/adbeef > actual && | 
 | 	test_cmp expect_non-note3 actual | 
 |  | 
 | ' | 
 |  | 
 |  | 
 | rm input expect | 
 | num_notes_refs=10 | 
 | num_commits=16 | 
 | some_commits=8 | 
 | # Create commits | 
 | fast_import_commits "refs/heads/more_commits" $num_commits input | 
 | # Create one note per above commit per notes ref | 
 | i=0 | 
 | while test $i -lt $num_notes_refs | 
 | do | 
 | 	i=$(($i + 1)) | 
 | 	fast_import_notes "refs/notes/more_notes_$i" $num_commits input | 
 | done | 
 | # Trigger branch reloading in git-fast-import by repeating the note creation | 
 | i=0 | 
 | while test $i -lt $num_notes_refs | 
 | do | 
 | 	i=$(($i + 1)) | 
 | 	fast_import_notes "refs/notes/more_notes_$i" $some_commits input " (2)" | 
 | done | 
 | # Finally create the expected output from the notes in refs/notes/more_notes_1 | 
 | i=$num_commits | 
 | while test $i -gt 0 | 
 | do | 
 | 	note_data="note for commit #$i" | 
 | 	if test $i -le $some_commits | 
 | 	then | 
 | 		note_data="$note_data (2)" | 
 | 	fi | 
 | 	cat >>expect <<EXPECT_END | 
 |     commit #$i | 
 |     $note_data | 
 | EXPECT_END | 
 | 	i=$(($i - 1)) | 
 | done | 
 |  | 
 | test_expect_success "add notes to $num_commits commits in each of $num_notes_refs refs" ' | 
 |  | 
 | 	git fast-import --active-branches=5 <input && | 
 | 	GIT_NOTES_REF=refs/notes/more_notes_1 git log refs/heads/more_commits | | 
 | 	    grep "^    " > actual && | 
 | 	test_cmp expect actual | 
 |  | 
 | ' | 
 |  | 
 | test_done |