The builtin-ification includes some minor behavioural changes to the
command-line interface: It is no longer allowed to mix the -m and -F
arguments, and it is not allowed to use multiple -F options.

As part of the builtin-ification, we add the commit_notes() function
to the builtin API. This function (together with the notes.h API) can
be easily used from other builtins to manipulate the notes tree.

Also includes needed changes to t3301.

This patch has been improved by the following contributions:
- Stephen Boyd: Use die() instead of fprintf(stderr, ...) followed by exit(1)

Cc: Stephen Boyd <bebarino@gmail.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:20 +01:00 коммит произвёл Junio C Hamano
Родитель 73f464b5f3
Коммит cd067d3bf4
7 изменённых файлов: 350 добавлений и 40 удалений

Просмотреть файл

@ -37,14 +37,12 @@ OPTIONS
------- -------
-m <msg>:: -m <msg>::
Use the given note message (instead of prompting). Use the given note message (instead of prompting).
If multiple `-m` (or `-F`) options are given, their If multiple `-m` options are given, their values
values are concatenated as separate paragraphs. are concatenated as separate paragraphs.
-F <file>:: -F <file>::
Take the note message from the given file. Use '-' to Take the note message from the given file. Use '-' to
read the note message from the standard input. read the note message from the standard input.
If multiple `-F` (or `-m`) options are given, their
values are concatenated as separate paragraphs.
Author Author

Просмотреть файл

@ -353,7 +353,6 @@ SCRIPT_SH += git-merge-one-file.sh
SCRIPT_SH += git-merge-resolve.sh SCRIPT_SH += git-merge-resolve.sh
SCRIPT_SH += git-mergetool.sh SCRIPT_SH += git-mergetool.sh
SCRIPT_SH += git-mergetool--lib.sh SCRIPT_SH += git-mergetool--lib.sh
SCRIPT_SH += git-notes.sh
SCRIPT_SH += git-parse-remote.sh SCRIPT_SH += git-parse-remote.sh
SCRIPT_SH += git-pull.sh SCRIPT_SH += git-pull.sh
SCRIPT_SH += git-quiltimport.sh SCRIPT_SH += git-quiltimport.sh
@ -671,6 +670,7 @@ BUILTIN_OBJS += builtin-mktag.o
BUILTIN_OBJS += builtin-mktree.o BUILTIN_OBJS += builtin-mktree.o
BUILTIN_OBJS += builtin-mv.o BUILTIN_OBJS += builtin-mv.o
BUILTIN_OBJS += builtin-name-rev.o BUILTIN_OBJS += builtin-name-rev.o
BUILTIN_OBJS += builtin-notes.o
BUILTIN_OBJS += builtin-pack-objects.o BUILTIN_OBJS += builtin-pack-objects.o
BUILTIN_OBJS += builtin-pack-redundant.o BUILTIN_OBJS += builtin-pack-redundant.o
BUILTIN_OBJS += builtin-pack-refs.o BUILTIN_OBJS += builtin-pack-refs.o

280
builtin-notes.c Normal file
Просмотреть файл

@ -0,0 +1,280 @@
/*
* Builtin "git notes"
*
* Copyright (c) 2010 Johan Herland <johan@herland.net>
*
* Based on git-notes.sh by Johannes Schindelin,
* and builtin-tag.c by Kristian Høgsberg and Carlos Rica.
*/
#include "cache.h"
#include "builtin.h"
#include "notes.h"
#include "blob.h"
#include "commit.h"
#include "refs.h"
#include "exec_cmd.h"
#include "run-command.h"
#include "parse-options.h"
static const char * const git_notes_usage[] = {
"git notes edit [-m <msg> | -F <file>] [<object>]",
"git notes show [<object>]",
NULL
};
static const char note_template[] =
"\n"
"#\n"
"# Write/edit the notes for the following object:\n"
"#\n";
static void write_note_data(int fd, const unsigned char *sha1)
{
unsigned long size;
enum object_type type;
char *buf = read_sha1_file(sha1, &type, &size);
if (buf) {
if (size)
write_or_die(fd, buf, size);
free(buf);
}
}
static void write_commented_object(int fd, const unsigned char *object)
{
const char *show_args[5] =
{"show", "--stat", "--no-notes", sha1_to_hex(object), NULL};
struct child_process show;
struct strbuf buf = STRBUF_INIT;
FILE *show_out;
/* Invoke "git show --stat --no-notes $object" */
memset(&show, 0, sizeof(show));
show.argv = show_args;
show.no_stdin = 1;
show.out = -1;
show.err = 0;
show.git_cmd = 1;
if (start_command(&show))
die("unable to start 'show' for object '%s'",
sha1_to_hex(object));
/* Open the output as FILE* so strbuf_getline() can be used. */
show_out = xfdopen(show.out, "r");
if (show_out == NULL)
die_errno("can't fdopen 'show' output fd");
/* Prepend "# " to each output line and write result to 'fd' */
while (strbuf_getline(&buf, show_out, '\n') != EOF) {
write_or_die(fd, "# ", 2);
write_or_die(fd, buf.buf, buf.len);
write_or_die(fd, "\n", 1);
}
strbuf_release(&buf);
if (fclose(show_out))
die_errno("failed to close pipe to 'show' for object '%s'",
sha1_to_hex(object));
if (finish_command(&show))
die("failed to finish 'show' for object '%s'",
sha1_to_hex(object));
}
static void create_note(const unsigned char *object,
struct strbuf *buf,
int skip_editor,
const unsigned char *prev,
unsigned char *result)
{
char *path = NULL;
if (!skip_editor) {
int fd;
/* write the template message before editing: */
path = git_pathdup("NOTES_EDITMSG");
fd = open(path, O_CREAT | O_TRUNC | O_WRONLY, 0600);
if (fd < 0)
die_errno("could not create file '%s'", path);
if (prev)
write_note_data(fd, prev);
write_or_die(fd, note_template, strlen(note_template));
write_commented_object(fd, object);
close(fd);
if (launch_editor(path, buf, NULL)) {
die("Please supply the note contents using either -m" \
" or -F option");
}
}
stripspace(buf, 1);
if (!skip_editor && !buf->len) {
fprintf(stderr, "Removing note for object %s\n",
sha1_to_hex(object));
hashclr(result);
} else {
if (write_sha1_file(buf->buf, buf->len, blob_type, result)) {
error("unable to write note object");
if (path)
error("The note contents has been left in %s",
path);
exit(128);
}
}
if (path) {
unlink_or_warn(path);
free(path);
}
}
struct msg_arg {
int given;
struct strbuf buf;
};
static int parse_msg_arg(const struct option *opt, const char *arg, int unset)
{
struct msg_arg *msg = opt->value;
if (!arg)
return -1;
if (msg->buf.len)
strbuf_addstr(&(msg->buf), "\n\n");
strbuf_addstr(&(msg->buf), arg);
msg->given = 1;
return 0;
}
int commit_notes(struct notes_tree *t, const char *msg)
{
struct commit_list *parent;
unsigned char tree_sha1[20], prev_commit[20], new_commit[20];
struct strbuf buf = STRBUF_INIT;
if (!t)
t = &default_notes_tree;
if (!t->initialized || !t->ref || !*t->ref)
die("Cannot commit uninitialized/unreferenced notes tree");
/* Prepare commit message and reflog message */
strbuf_addstr(&buf, "notes: "); /* commit message starts at index 7 */
strbuf_addstr(&buf, msg);
if (buf.buf[buf.len - 1] != '\n')
strbuf_addch(&buf, '\n'); /* Make sure msg ends with newline */
/* Convert notes tree to tree object */
if (write_notes_tree(t, tree_sha1))
die("Failed to write current notes tree to database");
/* Create new commit for the tree object */
if (!read_ref(t->ref, prev_commit)) { /* retrieve parent commit */
parent = xmalloc(sizeof(*parent));
parent->item = lookup_commit(prev_commit);
parent->next = NULL;
} else {
hashclr(prev_commit);
parent = NULL;
}
if (commit_tree(buf.buf + 7, tree_sha1, parent, new_commit, NULL))
die("Failed to commit notes tree to database");
/* Update notes ref with new commit */
update_ref(buf.buf, t->ref, new_commit, prev_commit, 0, DIE_ON_ERR);
strbuf_release(&buf);
return 0;
}
int cmd_notes(int argc, const char **argv, const char *prefix)
{
struct strbuf buf = STRBUF_INIT;
struct notes_tree *t;
unsigned char object[20], new_note[20];
const unsigned char *note;
const char *object_ref;
int edit = 0, show = 0;
const char *msgfile = NULL;
struct msg_arg msg = { 0, STRBUF_INIT };
struct option options[] = {
OPT_GROUP("Notes edit options"),
OPT_CALLBACK('m', NULL, &msg, "msg",
"note contents as a string", parse_msg_arg),
OPT_FILENAME('F', NULL, &msgfile, "note contents in a file"),
OPT_END()
};
git_config(git_default_config, NULL);
argc = parse_options(argc, argv, prefix, options, git_notes_usage, 0);
if (argc && !strcmp(argv[0], "edit"))
edit = 1;
else if (argc && !strcmp(argv[0], "show"))
show = 1;
if (edit + show != 1)
usage_with_options(git_notes_usage, options);
object_ref = argc == 2 ? argv[1] : "HEAD";
if (argc > 2) {
error("too many parameters");
usage_with_options(git_notes_usage, options);
}
if (get_sha1(object_ref, object))
die("Failed to resolve '%s' as a valid ref.", object_ref);
init_notes(NULL, NULL, NULL, 0);
t = &default_notes_tree;
if (prefixcmp(t->ref, "refs/notes/"))
die("Refusing to %s notes in %s (outside of refs/notes/)",
edit ? "edit" : "show", t->ref);
note = get_note(t, object);
/* show command */
if (show && !note) {
error("No note found for object %s.", sha1_to_hex(object));
return 1;
} else if (show) {
const char *show_args[3] = {"show", sha1_to_hex(note), NULL};
return execv_git_cmd(show_args);
}
/* edit command */
if (msg.given || msgfile) {
if (msg.given && msgfile) {
error("mixing -m and -F options is not allowed.");
usage_with_options(git_notes_usage, options);
}
if (msg.given)
strbuf_addbuf(&buf, &(msg.buf));
else {
if (!strcmp(msgfile, "-")) {
if (strbuf_read(&buf, 0, 1024) < 0)
die_errno("cannot read '%s'", msgfile);
} else {
if (strbuf_read_file(&buf, msgfile, 1024) < 0)
die_errno("could not open or read '%s'",
msgfile);
}
}
}
create_note(object, &buf, msg.given || msgfile, note, new_note);
add_note(t, object, new_note, combine_notes_overwrite);
commit_notes(t, "Note added by 'git notes edit'");
free_notes(t);
strbuf_release(&buf);
return 0;
}

Просмотреть файл

@ -5,6 +5,7 @@
#include "strbuf.h" #include "strbuf.h"
#include "cache.h" #include "cache.h"
#include "commit.h" #include "commit.h"
#include "notes.h"
extern const char git_version_string[]; extern const char git_version_string[];
extern const char git_usage_string[]; extern const char git_usage_string[];
@ -18,6 +19,7 @@ extern int fmt_merge_msg(int merge_summary, struct strbuf *in,
extern int commit_tree(const char *msg, unsigned char *tree, extern int commit_tree(const char *msg, unsigned char *tree,
struct commit_list *parents, unsigned char *ret, struct commit_list *parents, unsigned char *ret,
const char *author); const char *author);
extern int commit_notes(struct notes_tree *t, const char *msg);
extern int check_pager_config(const char *cmd); extern int check_pager_config(const char *cmd);
extern int cmd_add(int argc, const char **argv, const char *prefix); extern int cmd_add(int argc, const char **argv, const char *prefix);
@ -78,6 +80,7 @@ extern int cmd_mktag(int argc, const char **argv, const char *prefix);
extern int cmd_mktree(int argc, const char **argv, const char *prefix); extern int cmd_mktree(int argc, const char **argv, const char *prefix);
extern int cmd_mv(int argc, const char **argv, const char *prefix); extern int cmd_mv(int argc, const char **argv, const char *prefix);
extern int cmd_name_rev(int argc, const char **argv, const char *prefix); extern int cmd_name_rev(int argc, const char **argv, const char *prefix);
extern int cmd_notes(int argc, const char **argv, const char *prefix);
extern int cmd_pack_objects(int argc, const char **argv, const char *prefix); extern int cmd_pack_objects(int argc, const char **argv, const char *prefix);
extern int cmd_pack_redundant(int argc, const char **argv, const char *prefix); extern int cmd_pack_redundant(int argc, const char **argv, const char *prefix);
extern int cmd_patch_id(int argc, const char **argv, const char *prefix); extern int cmd_patch_id(int argc, const char **argv, const char *prefix);

Просмотреть файл

1
git.c
Просмотреть файл

@ -343,6 +343,7 @@ static void handle_internal_command(int argc, const char **argv)
{ "mktree", cmd_mktree, RUN_SETUP }, { "mktree", cmd_mktree, RUN_SETUP },
{ "mv", cmd_mv, RUN_SETUP | NEED_WORK_TREE }, { "mv", cmd_mv, RUN_SETUP | NEED_WORK_TREE },
{ "name-rev", cmd_name_rev, RUN_SETUP }, { "name-rev", cmd_name_rev, RUN_SETUP },
{ "notes", cmd_notes, RUN_SETUP },
{ "pack-objects", cmd_pack_objects, RUN_SETUP }, { "pack-objects", cmd_pack_objects, RUN_SETUP },
{ "pack-redundant", cmd_pack_redundant, RUN_SETUP }, { "pack-redundant", cmd_pack_redundant, RUN_SETUP },
{ "patch-id", cmd_patch_id }, { "patch-id", cmd_patch_id },

Просмотреть файл

@ -12,8 +12,8 @@ echo "$MSG" > "$1"
echo "$MSG" >& 2 echo "$MSG" >& 2
EOF EOF
chmod a+x fake_editor.sh chmod a+x fake_editor.sh
VISUAL=./fake_editor.sh GIT_EDITOR=./fake_editor.sh
export VISUAL 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 edit)
@ -56,8 +56,17 @@ 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 &&
test ! -f .git/NOTES_EDITMSG &&
test 1 = $(git ls-tree refs/notes/commits | wc -l) &&
test b0 = $(git notes show) &&
git show HEAD^ &&
test_must_fail git notes show HEAD^
'
test_expect_success 'edit existing notes' '
MSG=b1 git notes edit && MSG=b1 git notes edit &&
test ! -f .git/new-notes && 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) &&
git show HEAD^ && git show HEAD^ &&
@ -110,19 +119,16 @@ test_expect_success 'show multi-line notes' '
git log -2 > output && git log -2 > output &&
test_cmp expect-multiline output test_cmp expect-multiline output
' '
test_expect_success 'create -m and -F notes (setup)' ' test_expect_success 'create -F notes (setup)' '
: > a4 && : > a4 &&
git add a4 && git add a4 &&
test_tick && test_tick &&
git commit -m 4th && git commit -m 4th &&
echo "xyzzy" > note5 && echo "xyzzy" > note5 &&
git notes edit -m spam -F note5 -m "foo git notes edit -F note5
bar
baz"
' '
whitespace=" " cat > expect-F << EOF
cat > expect-m-and-F << EOF
commit 15023535574ded8b1a89052b32673f84cf9582b8 commit 15023535574ded8b1a89052b32673f84cf9582b8
Author: A U Thor <author@example.com> Author: A U Thor <author@example.com>
Date: Thu Apr 7 15:16:13 2005 -0700 Date: Thu Apr 7 15:16:13 2005 -0700
@ -130,21 +136,15 @@ Date: Thu Apr 7 15:16:13 2005 -0700
4th 4th
Notes: Notes:
spam
$whitespace
xyzzy xyzzy
$whitespace
foo
bar
baz
EOF EOF
printf "\n" >> expect-m-and-F printf "\n" >> expect-F
cat expect-multiline >> expect-m-and-F cat expect-multiline >> expect-F
test_expect_success 'show -m and -F notes' ' test_expect_success 'show -F notes' '
git log -3 > output && git log -3 > output &&
test_cmp expect-m-and-F output test_cmp expect-F output
' '
cat >expect << EOF cat >expect << EOF
@ -164,13 +164,7 @@ test_expect_success 'git log --pretty=raw does not show notes' '
cat >>expect <<EOF cat >>expect <<EOF
Notes: Notes:
spam
$whitespace
xyzzy xyzzy
$whitespace
foo
bar
baz
EOF EOF
test_expect_success 'git log --show-notes' ' test_expect_success 'git log --show-notes' '
git log -1 --pretty=raw --show-notes >output && git log -1 --pretty=raw --show-notes >output &&
@ -179,17 +173,17 @@ test_expect_success 'git log --show-notes' '
test_expect_success 'git log --no-notes' ' test_expect_success 'git log --no-notes' '
git log -1 --no-notes >output && git log -1 --no-notes >output &&
! grep spam output ! grep xyzzy output
' '
test_expect_success 'git format-patch does not show notes' ' test_expect_success 'git format-patch does not show notes' '
git format-patch -1 --stdout >output && git format-patch -1 --stdout >output &&
! grep spam output ! grep xyzzy output
' '
test_expect_success 'git format-patch --show-notes does show notes' ' test_expect_success 'git format-patch --show-notes does show notes' '
git format-patch --show-notes -1 --stdout >output && git format-patch --show-notes -1 --stdout >output &&
grep spam output grep xyzzy output
' '
for pretty in \ for pretty in \
@ -202,35 +196,69 @@ do
esac esac
test_expect_success "git show $pretty does$not show notes" ' test_expect_success "git show $pretty does$not show notes" '
git show $p >output && git show $p >output &&
eval "$negate grep spam output" eval "$negate grep xyzzy output"
' '
done done
test_expect_success 'create other note on a different notes ref (setup)' ' test_expect_success 'create -m notes (setup)' '
: > a5 && : > a5 &&
git add a5 && git add a5 &&
test_tick && test_tick &&
git commit -m 5th && git commit -m 5th &&
GIT_NOTES_REF="refs/notes/other" git notes edit -m "other note" git notes edit -m spam -m "foo
bar
baz"
' '
cat > expect-other << EOF whitespace=" "
cat > expect-m << EOF
commit bd1753200303d0a0344be813e504253b3d98e74d commit bd1753200303d0a0344be813e504253b3d98e74d
Author: A U Thor <author@example.com> Author: A U Thor <author@example.com>
Date: Thu Apr 7 15:17:13 2005 -0700 Date: Thu Apr 7 15:17:13 2005 -0700
5th 5th
Notes:
spam
$whitespace
foo
bar
baz
EOF
printf "\n" >> expect-m
cat expect-F >> expect-m
test_expect_success 'show -m notes' '
git log -4 > output &&
test_cmp expect-m output
'
test_expect_success 'create other note on a different notes ref (setup)' '
: > a6 &&
git add a6 &&
test_tick &&
git commit -m 6th &&
GIT_NOTES_REF="refs/notes/other" git notes edit -m "other note"
'
cat > expect-other << EOF
commit 387a89921c73d7ed72cd94d179c1c7048ca47756
Author: A U Thor <author@example.com>
Date: Thu Apr 7 15:18:13 2005 -0700
6th
Notes: Notes:
other note other note
EOF EOF
cat > expect-not-other << EOF cat > expect-not-other << EOF
commit bd1753200303d0a0344be813e504253b3d98e74d commit 387a89921c73d7ed72cd94d179c1c7048ca47756
Author: A U Thor <author@example.com> Author: A U Thor <author@example.com>
Date: Thu Apr 7 15:17:13 2005 -0700 Date: Thu Apr 7 15:18:13 2005 -0700
5th 6th
EOF EOF
test_expect_success 'Do not show note on other ref by default' ' test_expect_success 'Do not show note on other ref by default' '