Merge branch 'js/notes'
* js/notes: git-notes: fix printing of multi-line notes notes: fix core.notesRef documentation Add an expensive test for git-notes Speed up git notes lookup Add a script to edit/inspect notes Introduce commit notes Conflicts: pretty.c
This commit is contained in:
commit
7b75b331f6
1
.gitignore
vendored
1
.gitignore
vendored
@ -82,6 +82,7 @@ git-mktag
|
|||||||
git-mktree
|
git-mktree
|
||||||
git-name-rev
|
git-name-rev
|
||||||
git-mv
|
git-mv
|
||||||
|
git-notes
|
||||||
git-pack-redundant
|
git-pack-redundant
|
||||||
git-pack-objects
|
git-pack-objects
|
||||||
git-pack-refs
|
git-pack-refs
|
||||||
|
@ -422,6 +422,19 @@ relatively high IO latencies. With this set to 'true', git will do the
|
|||||||
index comparison to the filesystem data in parallel, allowing
|
index comparison to the filesystem data in parallel, allowing
|
||||||
overlapping IO's.
|
overlapping IO's.
|
||||||
|
|
||||||
|
core.notesRef::
|
||||||
|
When showing commit messages, also show notes which are stored in
|
||||||
|
the given ref. This ref is expected to contain files named
|
||||||
|
after the full SHA-1 of the commit they annotate.
|
||||||
|
+
|
||||||
|
If such a file exists in the given ref, the referenced blob is read, and
|
||||||
|
appended to the commit message, separated by a "Notes:" line. If the
|
||||||
|
given ref itself does not exist, it is not an error, but means that no
|
||||||
|
notes should be printed.
|
||||||
|
+
|
||||||
|
This setting defaults to "refs/notes/commits", and can be overridden by
|
||||||
|
the `GIT_NOTES_REF` environment variable.
|
||||||
|
|
||||||
alias.*::
|
alias.*::
|
||||||
Command aliases for the linkgit:git[1] command wrapper - e.g.
|
Command aliases for the linkgit:git[1] command wrapper - e.g.
|
||||||
after defining "alias.last = cat-file commit HEAD", the invocation
|
after defining "alias.last = cat-file commit HEAD", the invocation
|
||||||
|
46
Documentation/git-notes.txt
Normal file
46
Documentation/git-notes.txt
Normal file
@ -0,0 +1,46 @@
|
|||||||
|
git-notes(1)
|
||||||
|
============
|
||||||
|
|
||||||
|
NAME
|
||||||
|
----
|
||||||
|
git-notes - Add/inspect commit notes
|
||||||
|
|
||||||
|
SYNOPSIS
|
||||||
|
--------
|
||||||
|
[verse]
|
||||||
|
'git-notes' (edit | show) [commit]
|
||||||
|
|
||||||
|
DESCRIPTION
|
||||||
|
-----------
|
||||||
|
This command allows you to add notes to commit messages, without
|
||||||
|
changing the commit. To discern these notes from the message stored
|
||||||
|
in the commit object, the notes are indented like the message, after
|
||||||
|
an unindented line saying "Notes:".
|
||||||
|
|
||||||
|
To disable commit notes, you have to set the config variable
|
||||||
|
core.notesRef to the empty string. Alternatively, you can set it
|
||||||
|
to a different ref, something like "refs/notes/bugzilla". This setting
|
||||||
|
can be overridden by the environment variable "GIT_NOTES_REF".
|
||||||
|
|
||||||
|
|
||||||
|
SUBCOMMANDS
|
||||||
|
-----------
|
||||||
|
|
||||||
|
edit::
|
||||||
|
Edit the notes for a given commit (defaults to HEAD).
|
||||||
|
|
||||||
|
show::
|
||||||
|
Show the notes for a given commit (defaults to HEAD).
|
||||||
|
|
||||||
|
|
||||||
|
Author
|
||||||
|
------
|
||||||
|
Written by Johannes Schindelin <johannes.schindelin@gmx.de>
|
||||||
|
|
||||||
|
Documentation
|
||||||
|
-------------
|
||||||
|
Documentation by Johannes Schindelin
|
||||||
|
|
||||||
|
GIT
|
||||||
|
---
|
||||||
|
Part of the gitlink:git[7] suite
|
3
Makefile
3
Makefile
@ -265,6 +265,7 @@ SCRIPT_SH += git-merge-octopus.sh
|
|||||||
SCRIPT_SH += git-merge-one-file.sh
|
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-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
|
||||||
@ -377,6 +378,7 @@ LIB_H += ll-merge.h
|
|||||||
LIB_H += log-tree.h
|
LIB_H += log-tree.h
|
||||||
LIB_H += mailmap.h
|
LIB_H += mailmap.h
|
||||||
LIB_H += merge-recursive.h
|
LIB_H += merge-recursive.h
|
||||||
|
LIB_H += notes.h
|
||||||
LIB_H += object.h
|
LIB_H += object.h
|
||||||
LIB_H += pack.h
|
LIB_H += pack.h
|
||||||
LIB_H += pack-refs.h
|
LIB_H += pack-refs.h
|
||||||
@ -459,6 +461,7 @@ LIB_OBJS += match-trees.o
|
|||||||
LIB_OBJS += merge-file.o
|
LIB_OBJS += merge-file.o
|
||||||
LIB_OBJS += merge-recursive.o
|
LIB_OBJS += merge-recursive.o
|
||||||
LIB_OBJS += name-hash.o
|
LIB_OBJS += name-hash.o
|
||||||
|
LIB_OBJS += notes.o
|
||||||
LIB_OBJS += object.o
|
LIB_OBJS += object.o
|
||||||
LIB_OBJS += pack-check.o
|
LIB_OBJS += pack-check.o
|
||||||
LIB_OBJS += pack-refs.o
|
LIB_OBJS += pack-refs.o
|
||||||
|
3
cache.h
3
cache.h
@ -371,6 +371,8 @@ static inline enum object_type object_type(unsigned int mode)
|
|||||||
#define GITATTRIBUTES_FILE ".gitattributes"
|
#define GITATTRIBUTES_FILE ".gitattributes"
|
||||||
#define INFOATTRIBUTES_FILE "info/attributes"
|
#define INFOATTRIBUTES_FILE "info/attributes"
|
||||||
#define ATTRIBUTE_MACRO_PREFIX "[attr]"
|
#define ATTRIBUTE_MACRO_PREFIX "[attr]"
|
||||||
|
#define GIT_NOTES_REF_ENVIRONMENT "GIT_NOTES_REF"
|
||||||
|
#define GIT_NOTES_DEFAULT_REF "refs/notes/commits"
|
||||||
|
|
||||||
extern int is_bare_repository_cfg;
|
extern int is_bare_repository_cfg;
|
||||||
extern int is_bare_repository(void);
|
extern int is_bare_repository(void);
|
||||||
@ -542,6 +544,7 @@ enum rebase_setup_type {
|
|||||||
|
|
||||||
extern enum branch_track git_branch_track;
|
extern enum branch_track git_branch_track;
|
||||||
extern enum rebase_setup_type autorebase;
|
extern enum rebase_setup_type autorebase;
|
||||||
|
extern char *notes_ref_name;
|
||||||
|
|
||||||
#define GIT_REPO_VERSION 0
|
#define GIT_REPO_VERSION 0
|
||||||
extern int repository_format_version;
|
extern int repository_format_version;
|
||||||
|
@ -73,6 +73,7 @@ git-mktag plumbingmanipulators
|
|||||||
git-mktree plumbingmanipulators
|
git-mktree plumbingmanipulators
|
||||||
git-mv mainporcelain common
|
git-mv mainporcelain common
|
||||||
git-name-rev plumbinginterrogators
|
git-name-rev plumbinginterrogators
|
||||||
|
git-notes mainporcelain
|
||||||
git-pack-objects plumbingmanipulators
|
git-pack-objects plumbingmanipulators
|
||||||
git-pack-redundant plumbinginterrogators
|
git-pack-redundant plumbinginterrogators
|
||||||
git-pack-refs ancillarymanipulators
|
git-pack-refs ancillarymanipulators
|
||||||
|
1
commit.c
1
commit.c
@ -5,6 +5,7 @@
|
|||||||
#include "utf8.h"
|
#include "utf8.h"
|
||||||
#include "diff.h"
|
#include "diff.h"
|
||||||
#include "revision.h"
|
#include "revision.h"
|
||||||
|
#include "notes.h"
|
||||||
|
|
||||||
int save_commit_buffer = 1;
|
int save_commit_buffer = 1;
|
||||||
|
|
||||||
|
5
config.c
5
config.c
@ -469,6 +469,11 @@ static int git_default_core_config(const char *var, const char *value)
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!strcmp(var, "core.notesref")) {
|
||||||
|
notes_ref_name = xstrdup(value);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
if (!strcmp(var, "core.pager"))
|
if (!strcmp(var, "core.pager"))
|
||||||
return git_config_string(&pager_program, var, value);
|
return git_config_string(&pager_program, var, value);
|
||||||
|
|
||||||
|
@ -45,6 +45,7 @@ enum rebase_setup_type autorebase = AUTOREBASE_NEVER;
|
|||||||
|
|
||||||
/* Parallel index stat data preload? */
|
/* Parallel index stat data preload? */
|
||||||
int core_preload_index = 0;
|
int core_preload_index = 0;
|
||||||
|
char *notes_ref_name;
|
||||||
|
|
||||||
/* This is set by setup_git_dir_gently() and/or git_default_config() */
|
/* This is set by setup_git_dir_gently() and/or git_default_config() */
|
||||||
char *git_work_tree_cfg;
|
char *git_work_tree_cfg;
|
||||||
|
65
git-notes.sh
Executable file
65
git-notes.sh
Executable file
@ -0,0 +1,65 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
|
||||||
|
USAGE="(edit | show) [commit]"
|
||||||
|
. git-sh-setup
|
||||||
|
|
||||||
|
test -n "$3" && usage
|
||||||
|
|
||||||
|
test -z "$1" && usage
|
||||||
|
ACTION="$1"; shift
|
||||||
|
|
||||||
|
test -z "$GIT_NOTES_REF" && GIT_NOTES_REF="$(git config core.notesref)"
|
||||||
|
test -z "$GIT_NOTES_REF" && GIT_NOTES_REF="refs/notes/commits"
|
||||||
|
|
||||||
|
COMMIT=$(git rev-parse --verify --default HEAD "$@") ||
|
||||||
|
die "Invalid commit: $@"
|
||||||
|
|
||||||
|
MESSAGE="$GIT_DIR"/new-notes-$COMMIT
|
||||||
|
trap '
|
||||||
|
test -f "$MESSAGE" && rm "$MESSAGE"
|
||||||
|
' 0
|
||||||
|
|
||||||
|
case "$ACTION" in
|
||||||
|
edit)
|
||||||
|
GIT_NOTES_REF= git log -1 $COMMIT | sed "s/^/#/" > "$MESSAGE"
|
||||||
|
|
||||||
|
GIT_INDEX_FILE="$MESSAGE".idx
|
||||||
|
export GIT_INDEX_FILE
|
||||||
|
|
||||||
|
CURRENT_HEAD=$(git show-ref "$GIT_NOTES_REF" | cut -f 1 -d ' ')
|
||||||
|
if [ -z "$CURRENT_HEAD" ]; then
|
||||||
|
PARENT=
|
||||||
|
else
|
||||||
|
PARENT="-p $CURRENT_HEAD"
|
||||||
|
git read-tree "$GIT_NOTES_REF" || die "Could not read index"
|
||||||
|
git cat-file blob :$COMMIT >> "$MESSAGE" 2> /dev/null
|
||||||
|
fi
|
||||||
|
|
||||||
|
${VISUAL:-${EDITOR:-vi}} "$MESSAGE"
|
||||||
|
|
||||||
|
grep -v ^# < "$MESSAGE" | git stripspace > "$MESSAGE".processed
|
||||||
|
mv "$MESSAGE".processed "$MESSAGE"
|
||||||
|
if [ -s "$MESSAGE" ]; then
|
||||||
|
BLOB=$(git hash-object -w "$MESSAGE") ||
|
||||||
|
die "Could not write into object database"
|
||||||
|
git update-index --add --cacheinfo 0644 $BLOB $COMMIT ||
|
||||||
|
die "Could not write index"
|
||||||
|
else
|
||||||
|
test -z "$CURRENT_HEAD" &&
|
||||||
|
die "Will not initialise with empty tree"
|
||||||
|
git update-index --force-remove $COMMIT ||
|
||||||
|
die "Could not update index"
|
||||||
|
fi
|
||||||
|
|
||||||
|
TREE=$(git write-tree) || die "Could not write tree"
|
||||||
|
NEW_HEAD=$(echo Annotate $COMMIT | git commit-tree $TREE $PARENT) ||
|
||||||
|
die "Could not annotate"
|
||||||
|
git update-ref -m "Annotate $COMMIT" \
|
||||||
|
"$GIT_NOTES_REF" $NEW_HEAD $CURRENT_HEAD
|
||||||
|
;;
|
||||||
|
show)
|
||||||
|
git show "$GIT_NOTES_REF":$COMMIT
|
||||||
|
;;
|
||||||
|
*)
|
||||||
|
usage
|
||||||
|
esac
|
160
notes.c
Normal file
160
notes.c
Normal file
@ -0,0 +1,160 @@
|
|||||||
|
#include "cache.h"
|
||||||
|
#include "commit.h"
|
||||||
|
#include "notes.h"
|
||||||
|
#include "refs.h"
|
||||||
|
#include "utf8.h"
|
||||||
|
#include "strbuf.h"
|
||||||
|
#include "tree-walk.h"
|
||||||
|
|
||||||
|
struct entry {
|
||||||
|
unsigned char commit_sha1[20];
|
||||||
|
unsigned char notes_sha1[20];
|
||||||
|
};
|
||||||
|
|
||||||
|
struct hash_map {
|
||||||
|
struct entry *entries;
|
||||||
|
off_t count, size;
|
||||||
|
};
|
||||||
|
|
||||||
|
static int initialized;
|
||||||
|
static struct hash_map hash_map;
|
||||||
|
|
||||||
|
static int hash_index(struct hash_map *map, const unsigned char *sha1)
|
||||||
|
{
|
||||||
|
int i = ((*(unsigned int *)sha1) % map->size);
|
||||||
|
|
||||||
|
for (;;) {
|
||||||
|
unsigned char *current = map->entries[i].commit_sha1;
|
||||||
|
|
||||||
|
if (!hashcmp(sha1, current))
|
||||||
|
return i;
|
||||||
|
|
||||||
|
if (is_null_sha1(current))
|
||||||
|
return -1 - i;
|
||||||
|
|
||||||
|
if (++i == map->size)
|
||||||
|
i = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void add_entry(const unsigned char *commit_sha1,
|
||||||
|
const unsigned char *notes_sha1)
|
||||||
|
{
|
||||||
|
int index;
|
||||||
|
|
||||||
|
if (hash_map.count + 1 > hash_map.size >> 1) {
|
||||||
|
int i, old_size = hash_map.size;
|
||||||
|
struct entry *old = hash_map.entries;
|
||||||
|
|
||||||
|
hash_map.size = old_size ? old_size << 1 : 64;
|
||||||
|
hash_map.entries = (struct entry *)
|
||||||
|
xcalloc(sizeof(struct entry), hash_map.size);
|
||||||
|
|
||||||
|
for (i = 0; i < old_size; i++)
|
||||||
|
if (!is_null_sha1(old[i].commit_sha1)) {
|
||||||
|
index = -1 - hash_index(&hash_map,
|
||||||
|
old[i].commit_sha1);
|
||||||
|
memcpy(hash_map.entries + index, old + i,
|
||||||
|
sizeof(struct entry));
|
||||||
|
}
|
||||||
|
free(old);
|
||||||
|
}
|
||||||
|
|
||||||
|
index = hash_index(&hash_map, commit_sha1);
|
||||||
|
if (index < 0) {
|
||||||
|
index = -1 - index;
|
||||||
|
hash_map.count++;
|
||||||
|
}
|
||||||
|
|
||||||
|
hashcpy(hash_map.entries[index].commit_sha1, commit_sha1);
|
||||||
|
hashcpy(hash_map.entries[index].notes_sha1, notes_sha1);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void initialize_hash_map(const char *notes_ref_name)
|
||||||
|
{
|
||||||
|
unsigned char sha1[20], commit_sha1[20];
|
||||||
|
unsigned mode;
|
||||||
|
struct tree_desc desc;
|
||||||
|
struct name_entry entry;
|
||||||
|
void *buf;
|
||||||
|
|
||||||
|
if (!notes_ref_name || read_ref(notes_ref_name, commit_sha1) ||
|
||||||
|
get_tree_entry(commit_sha1, "", sha1, &mode))
|
||||||
|
return;
|
||||||
|
|
||||||
|
buf = fill_tree_descriptor(&desc, sha1);
|
||||||
|
if (!buf)
|
||||||
|
die("Could not read %s for notes-index", sha1_to_hex(sha1));
|
||||||
|
|
||||||
|
while (tree_entry(&desc, &entry))
|
||||||
|
if (!get_sha1(entry.path, commit_sha1))
|
||||||
|
add_entry(commit_sha1, entry.sha1);
|
||||||
|
free(buf);
|
||||||
|
}
|
||||||
|
|
||||||
|
static unsigned char *lookup_notes(const unsigned char *commit_sha1)
|
||||||
|
{
|
||||||
|
int index;
|
||||||
|
|
||||||
|
if (!hash_map.size)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
index = hash_index(&hash_map, commit_sha1);
|
||||||
|
if (index < 0)
|
||||||
|
return NULL;
|
||||||
|
return hash_map.entries[index].notes_sha1;
|
||||||
|
}
|
||||||
|
|
||||||
|
void get_commit_notes(const struct commit *commit, struct strbuf *sb,
|
||||||
|
const char *output_encoding)
|
||||||
|
{
|
||||||
|
static const char *utf8 = "utf-8";
|
||||||
|
unsigned char *sha1;
|
||||||
|
char *msg, *msg_p;
|
||||||
|
unsigned long linelen, msglen;
|
||||||
|
enum object_type type;
|
||||||
|
|
||||||
|
if (!initialized) {
|
||||||
|
const char *env = getenv(GIT_NOTES_REF_ENVIRONMENT);
|
||||||
|
if (env)
|
||||||
|
notes_ref_name = getenv(GIT_NOTES_REF_ENVIRONMENT);
|
||||||
|
else if (!notes_ref_name)
|
||||||
|
notes_ref_name = GIT_NOTES_DEFAULT_REF;
|
||||||
|
initialize_hash_map(notes_ref_name);
|
||||||
|
initialized = 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
sha1 = lookup_notes(commit->object.sha1);
|
||||||
|
if (!sha1)
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (!(msg = read_sha1_file(sha1, &type, &msglen)) || !msglen ||
|
||||||
|
type != OBJ_BLOB)
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (output_encoding && *output_encoding &&
|
||||||
|
strcmp(utf8, output_encoding)) {
|
||||||
|
char *reencoded = reencode_string(msg, output_encoding, utf8);
|
||||||
|
if (reencoded) {
|
||||||
|
free(msg);
|
||||||
|
msg = reencoded;
|
||||||
|
msglen = strlen(msg);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* we will end the annotation by a newline anyway */
|
||||||
|
if (msglen && msg[msglen - 1] == '\n')
|
||||||
|
msglen--;
|
||||||
|
|
||||||
|
strbuf_addstr(sb, "\nNotes:\n");
|
||||||
|
|
||||||
|
for (msg_p = msg; msg_p < msg + msglen; msg_p += linelen + 1) {
|
||||||
|
linelen = strchrnul(msg_p, '\n') - msg_p;
|
||||||
|
|
||||||
|
strbuf_addstr(sb, " ");
|
||||||
|
strbuf_add(sb, msg_p, linelen);
|
||||||
|
strbuf_addch(sb, '\n');
|
||||||
|
}
|
||||||
|
|
||||||
|
free(msg);
|
||||||
|
}
|
7
notes.h
Normal file
7
notes.h
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
#ifndef NOTES_H
|
||||||
|
#define NOTES_H
|
||||||
|
|
||||||
|
void get_commit_notes(const struct commit *commit, struct strbuf *sb,
|
||||||
|
const char *output_encoding);
|
||||||
|
|
||||||
|
#endif
|
5
pretty.c
5
pretty.c
@ -6,6 +6,7 @@
|
|||||||
#include "string-list.h"
|
#include "string-list.h"
|
||||||
#include "mailmap.h"
|
#include "mailmap.h"
|
||||||
#include "log-tree.h"
|
#include "log-tree.h"
|
||||||
|
#include "notes.h"
|
||||||
#include "color.h"
|
#include "color.h"
|
||||||
|
|
||||||
static char *user_format;
|
static char *user_format;
|
||||||
@ -920,5 +921,9 @@ void pretty_print_commit(enum cmit_fmt fmt, const struct commit *commit,
|
|||||||
*/
|
*/
|
||||||
if (fmt == CMIT_FMT_EMAIL && sb->len <= beginning_of_body)
|
if (fmt == CMIT_FMT_EMAIL && sb->len <= beginning_of_body)
|
||||||
strbuf_addch(sb, '\n');
|
strbuf_addch(sb, '\n');
|
||||||
|
|
||||||
|
if (fmt != CMIT_FMT_ONELINE)
|
||||||
|
get_commit_notes(commit, sb, encoding);
|
||||||
|
|
||||||
free(reencoded);
|
free(reencoded);
|
||||||
}
|
}
|
||||||
|
95
t/t3301-notes.sh
Executable file
95
t/t3301-notes.sh
Executable file
@ -0,0 +1,95 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
#
|
||||||
|
# Copyright (c) 2007 Johannes E. Schindelin
|
||||||
|
#
|
||||||
|
|
||||||
|
test_description='Test commit notes'
|
||||||
|
|
||||||
|
. ./test-lib.sh
|
||||||
|
|
||||||
|
cat > fake_editor.sh << \EOF
|
||||||
|
echo "$MSG" > "$1"
|
||||||
|
echo "$MSG" >& 2
|
||||||
|
EOF
|
||||||
|
chmod a+x fake_editor.sh
|
||||||
|
VISUAL=./fake_editor.sh
|
||||||
|
export VISUAL
|
||||||
|
|
||||||
|
test_expect_success 'cannot annotate non-existing HEAD' '
|
||||||
|
! MSG=3 git notes edit
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success setup '
|
||||||
|
: > a1 &&
|
||||||
|
git add a1 &&
|
||||||
|
test_tick &&
|
||||||
|
git commit -m 1st &&
|
||||||
|
: > a2 &&
|
||||||
|
git add a2 &&
|
||||||
|
test_tick &&
|
||||||
|
git commit -m 2nd
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success 'need valid notes ref' '
|
||||||
|
! MSG=1 GIT_NOTES_REF='/' git notes edit &&
|
||||||
|
! MSG=2 GIT_NOTES_REF='/' git notes show
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success 'create notes' '
|
||||||
|
git config core.notesRef refs/notes/commits &&
|
||||||
|
MSG=b1 git notes edit &&
|
||||||
|
test ! -f .git/new-notes &&
|
||||||
|
test 1 = $(git ls-tree refs/notes/commits | wc -l) &&
|
||||||
|
test b1 = $(git notes show) &&
|
||||||
|
git show HEAD^ &&
|
||||||
|
! git notes show HEAD^
|
||||||
|
'
|
||||||
|
|
||||||
|
cat > expect << EOF
|
||||||
|
commit 268048bfb8a1fb38e703baceb8ab235421bf80c5
|
||||||
|
Author: A U Thor <author@example.com>
|
||||||
|
Date: Thu Apr 7 15:14:13 2005 -0700
|
||||||
|
|
||||||
|
2nd
|
||||||
|
|
||||||
|
Notes:
|
||||||
|
b1
|
||||||
|
EOF
|
||||||
|
|
||||||
|
test_expect_success 'show notes' '
|
||||||
|
! (git cat-file commit HEAD | grep b1) &&
|
||||||
|
git log -1 > output &&
|
||||||
|
test_cmp expect output
|
||||||
|
'
|
||||||
|
test_expect_success 'create multi-line notes (setup)' '
|
||||||
|
: > a3 &&
|
||||||
|
git add a3 &&
|
||||||
|
test_tick &&
|
||||||
|
git commit -m 3rd &&
|
||||||
|
MSG="b3
|
||||||
|
c3c3c3c3
|
||||||
|
d3d3d3" git notes edit
|
||||||
|
'
|
||||||
|
|
||||||
|
cat > expect-multiline << EOF
|
||||||
|
commit 1584215f1d29c65e99c6c6848626553fdd07fd75
|
||||||
|
Author: A U Thor <author@example.com>
|
||||||
|
Date: Thu Apr 7 15:15:13 2005 -0700
|
||||||
|
|
||||||
|
3rd
|
||||||
|
|
||||||
|
Notes:
|
||||||
|
b3
|
||||||
|
c3c3c3c3
|
||||||
|
d3d3d3
|
||||||
|
EOF
|
||||||
|
|
||||||
|
printf "\n" >> expect-multiline
|
||||||
|
cat expect >> expect-multiline
|
||||||
|
|
||||||
|
test_expect_success 'show multi-line notes' '
|
||||||
|
git log -2 > output &&
|
||||||
|
test_cmp expect-multiline output
|
||||||
|
'
|
||||||
|
|
||||||
|
test_done
|
98
t/t3302-notes-index-expensive.sh
Executable file
98
t/t3302-notes-index-expensive.sh
Executable file
@ -0,0 +1,98 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
#
|
||||||
|
# Copyright (c) 2007 Johannes E. Schindelin
|
||||||
|
#
|
||||||
|
|
||||||
|
test_description='Test commit notes index (expensive!)'
|
||||||
|
|
||||||
|
. ./test-lib.sh
|
||||||
|
|
||||||
|
test -z "$GIT_NOTES_TIMING_TESTS" && {
|
||||||
|
say Skipping timing tests
|
||||||
|
test_done
|
||||||
|
exit
|
||||||
|
}
|
||||||
|
|
||||||
|
create_repo () {
|
||||||
|
number_of_commits=$1
|
||||||
|
nr=0
|
||||||
|
parent=
|
||||||
|
test -d .git || {
|
||||||
|
git init &&
|
||||||
|
tree=$(git write-tree) &&
|
||||||
|
while [ $nr -lt $number_of_commits ]; do
|
||||||
|
test_tick &&
|
||||||
|
commit=$(echo $nr | git commit-tree $tree $parent) ||
|
||||||
|
return
|
||||||
|
parent="-p $commit"
|
||||||
|
nr=$(($nr+1))
|
||||||
|
done &&
|
||||||
|
git update-ref refs/heads/master $commit &&
|
||||||
|
{
|
||||||
|
export GIT_INDEX_FILE=.git/temp;
|
||||||
|
git rev-list HEAD | cat -n | sed "s/^[ ][ ]*/ /g" |
|
||||||
|
while read nr sha1; do
|
||||||
|
blob=$(echo note $nr | git hash-object -w --stdin) &&
|
||||||
|
echo $sha1 | sed "s/^/0644 $blob 0 /"
|
||||||
|
done | git update-index --index-info &&
|
||||||
|
tree=$(git write-tree) &&
|
||||||
|
test_tick &&
|
||||||
|
commit=$(echo notes | git commit-tree $tree) &&
|
||||||
|
git update-ref refs/notes/commits $commit
|
||||||
|
} &&
|
||||||
|
git config core.notesRef refs/notes/commits
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
test_notes () {
|
||||||
|
count=$1 &&
|
||||||
|
git config core.notesRef refs/notes/commits &&
|
||||||
|
git log | grep "^ " > output &&
|
||||||
|
i=1 &&
|
||||||
|
while [ $i -le $count ]; do
|
||||||
|
echo " $(($count-$i))" &&
|
||||||
|
echo " note $i" &&
|
||||||
|
i=$(($i+1));
|
||||||
|
done > expect &&
|
||||||
|
git diff expect output
|
||||||
|
}
|
||||||
|
|
||||||
|
cat > time_notes << \EOF
|
||||||
|
mode=$1
|
||||||
|
i=1
|
||||||
|
while [ $i -lt $2 ]; do
|
||||||
|
case $1 in
|
||||||
|
no-notes)
|
||||||
|
export GIT_NOTES_REF=non-existing
|
||||||
|
;;
|
||||||
|
notes)
|
||||||
|
unset GIT_NOTES_REF
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
git log >/dev/null
|
||||||
|
i=$(($i+1))
|
||||||
|
done
|
||||||
|
EOF
|
||||||
|
|
||||||
|
time_notes () {
|
||||||
|
for mode in no-notes notes
|
||||||
|
do
|
||||||
|
echo $mode
|
||||||
|
/usr/bin/time sh ../time_notes $mode $1
|
||||||
|
done
|
||||||
|
}
|
||||||
|
|
||||||
|
for count in 10 100 1000 10000; do
|
||||||
|
|
||||||
|
mkdir $count
|
||||||
|
(cd $count;
|
||||||
|
|
||||||
|
test_expect_success "setup $count" "create_repo $count"
|
||||||
|
|
||||||
|
test_expect_success 'notes work' "test_notes $count"
|
||||||
|
|
||||||
|
test_expect_success 'notes timing' "time_notes 100"
|
||||||
|
)
|
||||||
|
done
|
||||||
|
|
||||||
|
test_done
|
Loading…
Reference in New Issue
Block a user