builtin-notes: Add "add" subcommand for adding notes to objects

"git notes add" is identical to "git notes edit" except that instead of
editing existing notes for a given object, you can only add notes to an
object that currently has none. If "git notes add" finds existing notes
for the given object, the addition is aborted. However, if the new
-f/--force option is used, "git notes add" will _overwrite_ the existing
notes with the new notes contents.

If there is no existing notes for the given object. "git notes add" is
identical to "git notes edit" (i.e. it adds a new note).

The patch includes tests verifying correct behaviour of the new subcommand.

Suggested-by: Joey Hess <joey@kitenet.net>
Improved-by: Junio C Hamano <gitster@pobox.com>
Signed-off-by: Johan Herland <johan@herland.net>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
This commit is contained in:
Johan Herland 2010-02-13 22:28:32 +01:00 committed by Junio C Hamano
parent ba20f15e0a
commit 7aa4754e55
3 changed files with 74 additions and 22 deletions

View File

@ -9,6 +9,7 @@ SYNOPSIS
-------- --------
[verse] [verse]
'git notes' [list [<object>]] 'git notes' [list [<object>]]
'git notes' add [-f] [-F <file> | -m <msg>] [<object>]
'git notes' edit [-F <file> | -m <msg>] [<object>] 'git notes' edit [-F <file> | -m <msg>] [<object>]
'git notes' show [<object>] 'git notes' show [<object>]
'git notes' remove [<object>] 'git notes' remove [<object>]
@ -41,6 +42,11 @@ list::
annotate (in the format "<note object> <annotated object>"). annotate (in the format "<note object> <annotated object>").
This is the default subcommand if no subcommand is given. This is the default subcommand if no subcommand is given.
add::
Add notes for a given object (defaults to HEAD). Abort if the
object already has notes, abort. (use `-f` to overwrite an
existing note).
edit:: edit::
Edit the notes for a given object (defaults to HEAD). Edit the notes for a given object (defaults to HEAD).
@ -57,6 +63,11 @@ prune::
OPTIONS OPTIONS
------- -------
-f::
--force::
When adding notes to an object that already has notes,
overwrite the existing notes (instead of aborting).
-m <msg>:: -m <msg>::
--message=<msg>:: --message=<msg>::
Use the given note message (instead of prompting). Use the given note message (instead of prompting).

View File

@ -19,6 +19,7 @@
static const char * const git_notes_usage[] = { static const char * const git_notes_usage[] = {
"git notes [list [<object>]]", "git notes [list [<object>]]",
"git notes add [-f] [-m <msg> | -F <file>] [<object>]",
"git notes edit [-m <msg> | -F <file>] [<object>]", "git notes edit [-m <msg> | -F <file>] [<object>]",
"git notes show [<object>]", "git notes show [<object>]",
"git notes remove [<object>]", "git notes remove [<object>]",
@ -211,7 +212,8 @@ int cmd_notes(int argc, const char **argv, const char *prefix)
const char *object_ref; const char *object_ref;
char logmsg[100]; char logmsg[100];
int list = 0, edit = 0, show = 0, remove = 0, prune = 0; int list = 0, add = 0, edit = 0, show = 0, remove = 0, prune = 0,
force = 0;
int given_object; int given_object;
const char *msgfile = NULL; const char *msgfile = NULL;
struct msg_arg msg = { 0, STRBUF_INIT }; struct msg_arg msg = { 0, STRBUF_INIT };
@ -220,6 +222,7 @@ int cmd_notes(int argc, const char **argv, const char *prefix)
OPT_CALLBACK('m', "message", &msg, "msg", OPT_CALLBACK('m', "message", &msg, "msg",
"note contents as a string", parse_msg_arg), "note contents as a string", parse_msg_arg),
OPT_FILENAME('F', "file", &msgfile, "note contents in a file"), OPT_FILENAME('F', "file", &msgfile, "note contents in a file"),
OPT_BOOLEAN('f', "force", &force, "replace existing notes"),
OPT_END() OPT_END()
}; };
@ -229,6 +232,8 @@ int cmd_notes(int argc, const char **argv, const char *prefix)
if (argc && !strcmp(argv[0], "list")) if (argc && !strcmp(argv[0], "list"))
list = 1; list = 1;
else if (argc && !strcmp(argv[0], "add"))
add = 1;
else if (argc && !strcmp(argv[0], "edit")) else if (argc && !strcmp(argv[0], "edit"))
edit = 1; edit = 1;
else if (argc && !strcmp(argv[0], "show")) else if (argc && !strcmp(argv[0], "show"))
@ -240,10 +245,10 @@ int cmd_notes(int argc, const char **argv, const char *prefix)
else if (!argc) else if (!argc)
list = 1; /* Default to 'list' if no other subcommand given */ list = 1; /* Default to 'list' if no other subcommand given */
if (list + edit + show + remove + prune != 1) if (list + add + edit + show + remove + prune != 1)
usage_with_options(git_notes_usage, options); usage_with_options(git_notes_usage, options);
if ((msg.given || msgfile) && !edit) { if ((msg.given || msgfile) && !(add || edit)) {
error("cannot use -m/-F options with %s subcommand.", argv[0]); error("cannot use -m/-F options with %s subcommand.", argv[0]);
usage_with_options(git_notes_usage, options); usage_with_options(git_notes_usage, options);
} }
@ -253,6 +258,11 @@ int cmd_notes(int argc, const char **argv, const char *prefix)
usage_with_options(git_notes_usage, options); usage_with_options(git_notes_usage, options);
} }
if (force && !add) {
error("cannot use -f option with %s subcommand.", argv[0]);
usage_with_options(git_notes_usage, options);
}
given_object = argc == 2; given_object = argc == 2;
object_ref = given_object ? argv[1] : "HEAD"; object_ref = given_object ? argv[1] : "HEAD";
if (argc > 2 || (prune && argc > 1)) { if (argc > 2 || (prune && argc > 1)) {
@ -294,7 +304,19 @@ int cmd_notes(int argc, const char **argv, const char *prefix)
return execv_git_cmd(show_args); return execv_git_cmd(show_args);
} }
/* edit/remove/prune command */ /* add/edit/remove/prune command */
if (add && note) {
if (force)
fprintf(stderr, "Overwriting existing notes for object %s\n",
sha1_to_hex(object));
else {
error("Cannot add notes. Found existing notes for object %s. "
"Use '-f' to overwrite existing notes",
sha1_to_hex(object));
return 1;
}
}
if (remove) if (remove)
strbuf_reset(&buf); strbuf_reset(&buf);

View File

@ -16,7 +16,7 @@ GIT_EDITOR=./fake_editor.sh
export GIT_EDITOR export GIT_EDITOR
test_expect_success 'cannot annotate non-existing HEAD' ' test_expect_success 'cannot annotate non-existing HEAD' '
(MSG=3 && export MSG && test_must_fail git notes edit) (MSG=3 && export MSG && test_must_fail git notes add)
' '
test_expect_success setup ' test_expect_success setup '
@ -32,18 +32,18 @@ test_expect_success setup '
test_expect_success 'need valid notes ref' ' test_expect_success 'need valid notes ref' '
(MSG=1 GIT_NOTES_REF=/ && export MSG GIT_NOTES_REF && (MSG=1 GIT_NOTES_REF=/ && export MSG GIT_NOTES_REF &&
test_must_fail git notes edit) && test_must_fail git notes add) &&
(MSG=2 GIT_NOTES_REF=/ && export MSG GIT_NOTES_REF && (MSG=2 GIT_NOTES_REF=/ && export MSG GIT_NOTES_REF &&
test_must_fail git notes show) test_must_fail git notes show)
' '
test_expect_success 'refusing to edit in refs/heads/' ' test_expect_success 'refusing to add notes in refs/heads/' '
(MSG=1 GIT_NOTES_REF=refs/heads/bogus && (MSG=1 GIT_NOTES_REF=refs/heads/bogus &&
export MSG GIT_NOTES_REF && export MSG GIT_NOTES_REF &&
test_must_fail git notes edit) test_must_fail git notes add)
' '
test_expect_success 'refusing to edit in refs/remotes/' ' test_expect_success 'refusing to edit notes in refs/remotes/' '
(MSG=1 GIT_NOTES_REF=refs/remotes/bogus && (MSG=1 GIT_NOTES_REF=refs/remotes/bogus &&
export MSG GIT_NOTES_REF && export MSG GIT_NOTES_REF &&
test_must_fail git notes edit) test_must_fail git notes edit)
@ -56,16 +56,34 @@ test_expect_success 'handle empty notes gracefully' '
test_expect_success 'create notes' ' test_expect_success 'create notes' '
git config core.notesRef refs/notes/commits && git config core.notesRef refs/notes/commits &&
MSG=b0 git notes edit && MSG=b4 git notes add &&
test ! -f .git/NOTES_EDITMSG && test ! -f .git/NOTES_EDITMSG &&
test 1 = $(git ls-tree refs/notes/commits | wc -l) && test 1 = $(git ls-tree refs/notes/commits | wc -l) &&
test b0 = $(git notes show) && test b4 = $(git notes show) &&
git show HEAD^ && git show HEAD^ &&
test_must_fail git notes show HEAD^ test_must_fail git notes show HEAD^
' '
test_expect_success 'edit existing notes' ' test_expect_success 'edit existing notes' '
MSG=b1 git notes edit && MSG=b3 git notes edit &&
test ! -f .git/NOTES_EDITMSG &&
test 1 = $(git ls-tree refs/notes/commits | wc -l) &&
test b3 = $(git notes show) &&
git show HEAD^ &&
test_must_fail git notes show HEAD^
'
test_expect_success 'cannot add note where one exists' '
! MSG=b2 git notes add &&
test ! -f .git/NOTES_EDITMSG &&
test 1 = $(git ls-tree refs/notes/commits | wc -l) &&
test b3 = $(git notes show) &&
git show HEAD^ &&
test_must_fail git notes show HEAD^
'
test_expect_success 'can overwrite existing note with "git notes add -f"' '
MSG=b1 git notes add -f &&
test ! -f .git/NOTES_EDITMSG && test ! -f .git/NOTES_EDITMSG &&
test 1 = $(git ls-tree refs/notes/commits | wc -l) && test 1 = $(git ls-tree refs/notes/commits | wc -l) &&
test b1 = $(git notes show) && test b1 = $(git notes show) &&
@ -89,6 +107,7 @@ test_expect_success 'show notes' '
git log -1 > output && git log -1 > output &&
test_cmp expect output test_cmp expect output
' '
test_expect_success 'create multi-line notes (setup)' ' test_expect_success 'create multi-line notes (setup)' '
: > a3 && : > a3 &&
git add a3 && git add a3 &&
@ -96,7 +115,7 @@ test_expect_success 'create multi-line notes (setup)' '
git commit -m 3rd && git commit -m 3rd &&
MSG="b3 MSG="b3
c3c3c3c3 c3c3c3c3
d3d3d3" git notes edit d3d3d3" git notes add
' '
cat > expect-multiline << EOF cat > expect-multiline << EOF
@ -125,7 +144,7 @@ test_expect_success 'create -F notes (setup)' '
test_tick && test_tick &&
git commit -m 4th && git commit -m 4th &&
echo "xyzzy" > note5 && echo "xyzzy" > note5 &&
git notes edit -F note5 git notes add -F note5
' '
cat > expect-F << EOF cat > expect-F << EOF
@ -205,7 +224,7 @@ test_expect_success 'create -m notes (setup)' '
git add a5 && git add a5 &&
test_tick && test_tick &&
git commit -m 5th && git commit -m 5th &&
git notes edit -m spam -m "foo git notes add -m spam -m "foo
bar bar
baz" baz"
' '
@ -234,8 +253,8 @@ test_expect_success 'show -m notes' '
test_cmp expect-m output test_cmp expect-m output
' '
test_expect_success 'remove note with -F /dev/null (setup)' ' test_expect_success 'remove note with add -f -F /dev/null (setup)' '
git notes edit -F /dev/null git notes add -f -F /dev/null
' '
cat > expect-rm-F << EOF cat > expect-rm-F << EOF
@ -256,7 +275,7 @@ test_expect_success 'verify note removal with -F /dev/null' '
' '
test_expect_success 'do not create empty note with -m "" (setup)' ' test_expect_success 'do not create empty note with -m "" (setup)' '
git notes edit -m "" git notes add -m ""
' '
test_expect_success 'verify non-creation of note with -m ""' ' test_expect_success 'verify non-creation of note with -m ""' '
@ -329,7 +348,7 @@ test_expect_success 'create other note on a different notes ref (setup)' '
git add a6 && git add a6 &&
test_tick && test_tick &&
git commit -m 6th && git commit -m 6th &&
GIT_NOTES_REF="refs/notes/other" git notes edit -m "other note" GIT_NOTES_REF="refs/notes/other" git notes add -m "other note"
' '
cat > expect-other << EOF cat > expect-other << EOF
@ -374,17 +393,17 @@ test_expect_success 'Do not show note when core.notesRef is overridden' '
test_expect_success 'Allow notes on non-commits (trees, blobs, tags)' ' test_expect_success 'Allow notes on non-commits (trees, blobs, tags)' '
echo "Note on a tree" > expect echo "Note on a tree" > expect
git notes edit -m "Note on a tree" HEAD: && git notes add -m "Note on a tree" HEAD: &&
git notes show HEAD: > actual && git notes show HEAD: > actual &&
test_cmp expect actual && test_cmp expect actual &&
echo "Note on a blob" > expect echo "Note on a blob" > expect
filename=$(git ls-tree --name-only HEAD | head -n1) && filename=$(git ls-tree --name-only HEAD | head -n1) &&
git notes edit -m "Note on a blob" HEAD:$filename && git notes add -m "Note on a blob" HEAD:$filename &&
git notes show HEAD:$filename > actual && git notes show HEAD:$filename > actual &&
test_cmp expect actual && test_cmp expect actual &&
echo "Note on a tag" > expect echo "Note on a tag" > expect
git tag -a -m "This is an annotated tag" foobar HEAD^ && git tag -a -m "This is an annotated tag" foobar HEAD^ &&
git notes edit -m "Note on a tag" foobar && git notes add -m "Note on a tag" foobar &&
git notes show foobar > actual && git notes show foobar > actual &&
test_cmp expect actual test_cmp expect actual
' '