From mboxrd@z Thu Jan 1 00:00:00 1970 From: Johan Herland Subject: [PATCH 2/2] notes: Add --allow-empty, to allow storing empty notes Date: Wed, 5 Nov 2014 02:32:55 +0100 Message-ID: <1415151175-1682-2-git-send-email-johan@herland.net> References: <1415151175-1682-1-git-send-email-johan@herland.net> Cc: gitster@pobox.com, mackyle@gmail.com, jhf@trifork.com, Johan Herland To: git@vger.kernel.org X-From: git-owner@vger.kernel.org Wed Nov 05 02:33:12 2014 Return-path: Envelope-to: gcvg-git-2@plane.gmane.org Received: from vger.kernel.org ([209.132.180.67]) by plane.gmane.org with esmtp (Exim 4.69) (envelope-from ) id 1XlpTD-000896-Ce for gcvg-git-2@plane.gmane.org; Wed, 05 Nov 2014 02:33:11 +0100 Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1752588AbaKEBdH (ORCPT ); Tue, 4 Nov 2014 20:33:07 -0500 Received: from mail-lb0-f176.google.com ([209.85.217.176]:46981 "EHLO mail-lb0-f176.google.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1752016AbaKEBdF (ORCPT ); Tue, 4 Nov 2014 20:33:05 -0500 Received: by mail-lb0-f176.google.com with SMTP id 10so1228225lbg.7 for ; Tue, 04 Nov 2014 17:33:03 -0800 (PST) X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20130820; h=from:to:cc:subject:date:message-id:in-reply-to:references; bh=DguF0NH6QLfbZs05jL+uF0d2anMc1OXG6iXgCYSoPj4=; b=XYZ0bH+mTadD2xNe6Mby2hlINuJ2VkIv4lOjtdJ+YrSiz26dA9jV7grRDergZkA+rZ muUsiyHRGzld3yN7eLoj+PhVE9wku9DGhpxNEaWD9hyQO9nigeiiwPPZzP/Hg95lDcU7 ldnG1kiRREXbFOnIy/Edp4rW5VeaPDrA1AwX4ly3sdL6GC/87JX+71z1DeH4XBx4oZKE vONUX/ZIoviOPu5kYqOSi5GNCpJF1x9V6nj3OE9dLGbR+l7ORzPQ87IRvmufEWgSJtli ZboXE0MlkK2RJiIm64e/cDWviwbJ9YMlQO8JbF5W1DFGY+O1bJXA72Eryjl80bqPg8Hp 9zPg== X-Received: by 10.112.54.229 with SMTP id m5mr63491542lbp.11.1415151183669; Tue, 04 Nov 2014 17:33:03 -0800 (PST) Received: from alpha.herland (245.37-191-128.fiber.lynet.no. [37.191.128.245]) by mx.google.com with ESMTPSA id q1sm745005lal.6.2014.11.04.17.33.01 for (version=TLSv1.2 cipher=ECDHE-RSA-AES128-SHA bits=128/128); Tue, 04 Nov 2014 17:33:02 -0800 (PST) X-Mailer: git-send-email 2.0.0.rc4.501.gdaf83ca In-Reply-To: <1415151175-1682-1-git-send-email-johan@herland.net> Sender: git-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: git@vger.kernel.org Although the "git notes" man page advertises that we support binary-safe notes addition (using the -C option), we currently do not support adding the empty note (i.e. using the empty blob to annotate an object). Instead, an empty note is always treated as an intent to remove the note altogether. Introduce the --allow-empty option to the add/append/edit subcommands, to explicitly allow an empty note to be stored into the notes tree. Also update the documentation, and add test cases for the new option. Reported-by: James H. Fisher Improved-by: Kyle J. McKay Signed-off-by: Johan Herland --- Documentation/git-notes.txt | 12 ++++++++---- builtin/notes.c | 25 +++++++++++++++---------- notes.c | 3 +-- t/t3312-notes-empty.sh | 20 +++++++++++++++++++- 4 files changed, 43 insertions(+), 17 deletions(-) diff --git a/Documentation/git-notes.txt b/Documentation/git-notes.txt index 310f0a5..851518d 100644 --- a/Documentation/git-notes.txt +++ b/Documentation/git-notes.txt @@ -9,10 +9,10 @@ SYNOPSIS -------- [verse] 'git notes' [list []] -'git notes' add [-f] [-F | -m | (-c | -C) ] [] +'git notes' add [-f] [--allow-empty] [-F | -m | (-c | -C) ] [] 'git notes' copy [-f] ( --stdin | ) -'git notes' append [-F | -m | (-c | -C) ] [] -'git notes' edit [] +'git notes' append [--allow-empty] [-F | -m | (-c | -C) ] [] +'git notes' edit [--allow-empty] [] 'git notes' show [] 'git notes' merge [-v | -q] [-s ] 'git notes' merge --commit [-v | -q] @@ -155,6 +155,10 @@ OPTIONS Like '-C', but with '-c' the editor is invoked, so that the user can further edit the note message. +--allow-empty:: + Allow an empty note object to be stored. The default behavior is + to automatically remove empty notes. + --ref :: Manipulate the notes tree in . This overrides 'GIT_NOTES_REF' and the "core.notesRef" configuration. The ref @@ -287,7 +291,7 @@ arbitrary files using 'git hash-object': ------------ $ cc *.c $ blob=$(git hash-object -w a.out) -$ git notes --ref=built add -C "$blob" HEAD +$ git notes --ref=built add --allow-empty -C "$blob" HEAD ------------ (You cannot simply use `git notes --ref=built add -F a.out HEAD` diff --git a/builtin/notes.c b/builtin/notes.c index 68b6cd8..038a419 100644 --- a/builtin/notes.c +++ b/builtin/notes.c @@ -22,10 +22,10 @@ static const char * const git_notes_usage[] = { N_("git notes [--ref ] [list []]"), - N_("git notes [--ref ] add [-f] [-m | -F | (-c | -C) ] []"), + N_("git notes [--ref ] add [-f] [--allow-empty] [-m | -F | (-c | -C) ] []"), N_("git notes [--ref ] copy [-f] "), - N_("git notes [--ref ] append [-m | -F | (-c | -C) ] []"), - N_("git notes [--ref ] edit []"), + N_("git notes [--ref ] append [--allow-empty] [-m | -F | (-c | -C) ] []"), + N_("git notes [--ref ] edit [--allow-empty] []"), N_("git notes [--ref ] show []"), N_("git notes [--ref ] merge [-v | -q] [-s ] "), N_("git notes merge --commit [-v | -q]"), @@ -150,8 +150,8 @@ static void write_commented_object(int fd, const unsigned char *object) } static void create_note(const unsigned char *object, struct msg_arg *msg, - int append_only, const unsigned char *prev, - unsigned char *result) + int append_only, int allow_empty, + const unsigned char *prev, unsigned char *result) { char *path = NULL; @@ -202,7 +202,7 @@ static void create_note(const unsigned char *object, struct msg_arg *msg, free(prev_buf); } - if (!msg->buf.len) { + if (!allow_empty && !msg->buf.len) { fprintf(stderr, _("Removing note for object %s\n"), sha1_to_hex(object)); hashclr(result); @@ -266,7 +266,7 @@ static int parse_reuse_arg(const struct option *opt, const char *arg, int unset) if (get_sha1(arg, object)) die(_("Failed to resolve '%s' as a valid ref."), arg); - if (!(buf = read_sha1_file(object, &type, &len)) || !len) { + if (!(buf = read_sha1_file(object, &type, &len))) { free(buf); die(_("Failed to read object '%s'."), arg); } @@ -397,7 +397,7 @@ static int append_edit(int argc, const char **argv, const char *prefix); static int add(int argc, const char **argv, const char *prefix) { - int retval = 0, force = 0; + int retval = 0, force = 0, allow_empty = 0; const char *object_ref; struct notes_tree *t; unsigned char object[20], new_note[20]; @@ -417,6 +417,8 @@ static int add(int argc, const char **argv, const char *prefix) { OPTION_CALLBACK, 'C', "reuse-message", &msg, N_("object"), N_("reuse specified note object"), PARSE_OPT_NONEG, parse_reuse_arg}, + OPT_BOOL(0, "allow-empty", &allow_empty, + N_("allow storing empty note")), OPT__FORCE(&force, N_("replace existing notes")), OPT_END() }; @@ -460,7 +462,7 @@ static int add(int argc, const char **argv, const char *prefix) sha1_to_hex(object)); } - create_note(object, &msg, 0, note, new_note); + create_note(object, &msg, 0, allow_empty, note, new_note); if (is_null_sha1(new_note)) remove_note(t, object); @@ -554,6 +556,7 @@ out: static int append_edit(int argc, const char **argv, const char *prefix) { + int allow_empty = 0; const char *object_ref; struct notes_tree *t; unsigned char object[20], new_note[20]; @@ -574,6 +577,8 @@ static int append_edit(int argc, const char **argv, const char *prefix) { OPTION_CALLBACK, 'C', "reuse-message", &msg, N_("object"), N_("reuse specified note object"), PARSE_OPT_NONEG, parse_reuse_arg}, + OPT_BOOL(0, "allow-empty", &allow_empty, + N_("allow storing empty note")), OPT_END() }; int edit = !strcmp(argv[0], "edit"); @@ -600,7 +605,7 @@ static int append_edit(int argc, const char **argv, const char *prefix) t = init_notes_check(argv[0]); note = get_note(t, object); - create_note(object, &msg, !edit, note, new_note); + create_note(object, &msg, !edit, allow_empty, note, new_note); if (is_null_sha1(new_note)) remove_note(t, object); diff --git a/notes.c b/notes.c index 5fe691d..62bc6e1 100644 --- a/notes.c +++ b/notes.c @@ -1218,8 +1218,7 @@ static void format_note(struct notes_tree *t, const unsigned char *object_sha1, if (!sha1) return; - if (!(msg = read_sha1_file(sha1, &type, &msglen)) || !msglen || - type != OBJ_BLOB) { + if (!(msg = read_sha1_file(sha1, &type, &msglen)) || type != OBJ_BLOB) { free(msg); return; } diff --git a/t/t3312-notes-empty.sh b/t/t3312-notes-empty.sh index 2806d27..f89fbc9 100755 --- a/t/t3312-notes-empty.sh +++ b/t/t3312-notes-empty.sh @@ -1,6 +1,6 @@ #!/bin/sh -test_description='Test adding/editing of empty notes' +test_description='Test adding/editing of empty notes with/without --allow-empty' . ./test-lib.sh cat >fake_editor.sh <<\EOF @@ -35,6 +35,18 @@ verify_missing() { ! git notes list HEAD } +cp expect_missing expect_empty +cat >>expect_empty <<\EOF + +Notes: +EOF + +verify_empty() { + git log -1 > actual && + test_cmp expect_empty actual && + test "$(git notes list HEAD)" = "$empty_blob" +} + for cmd in \ 'add' \ 'add -F /dev/null' \ @@ -53,6 +65,12 @@ do MSG= git notes $cmd && verify_missing " + + test_expect_success "'git notes $cmd --allow-empty' stores empty note" " + cleanup_notes && + MSG= git notes $cmd --allow-empty && + verify_empty + " done test_done -- 2.0.0.rc4.501.gdaf83ca