20

试图确定用户在使用 git-1.7.4.1 的对象数据库中被警告的速度有多快,我拉了一个一位 switcheroo:

$ git init 存储库
在 /tmp/repo/.git/ 中初始化空的 Git 存储库
$ cd 回购
$ echo '非常重要的信息' >关键
$ git 添加关键
$ git commit -m 关键
[master (root-commit) c4d6d90] 关键
 1 个文件已更改,1 个插入(+),0 个删除(-)
 创建模式 100644 关键
$ git ls-tree 头
100644 blob 82d423c32c4bb2c52938088e0234db041bf4eaaf 严重
$ git show 82d423c32c4bb2c52938088e0234db041bf4eaaf
非常重要的信息
$ echo '非常重要的信息' | git hash-object --stdin -w
81a3797afe76d339db25c0f9c705a6caa47279c2
$ mv .git/objects/81/a3797afe76d339db25c0f9c705a6caa47279c2 \
     .git/objects/82/d423c32c4bb2c52938088e0234db041bf4eaaf

当然,git-fsck通知

$ git fsck
错误:sha1 不匹配 82d423c32c4bb2c52938088e0234db041bf4eaaf

错误:82d423c32c4bb2c52938088e0234db041bf4eaaf:对象损坏或丢失
缺少 blob 82d423c32c4bb2c52938088e0234db041bf4eaaf

git-log对改变感到满意

$ 混帐日志 -p
提交 c4d6d90467af9ffa94772795d5c5d191228933c1
作者:格雷格·培根 <gbacon@dbresearch.net>
日期:2011 年 4 月 7 日星期四 12:20:53 -0500

    批判的

差异 --git a/关键 b/关键
新文件模式 100644
索引 0000000..82d423c
--- /dev/null
+++ b/关键
@@ -0,0 +1 @@
+非常重要的信息

原样git-checkout

$ rm 关键
$ 混帐结帐。
$ cat 关键
非常重要的信息

一个特定的调用git-show揭示了腐败

$ git show 82d423c32c4bb2c52938088e0234db041bf4eaaf
错误:sha1 不匹配 82d423c32c4bb2c52938088e0234db041bf4eaaf

致命:坏对象 82d423c32c4bb2c52938088e0234db041bf4eaaf

但不是更广泛的。

$ git 显示
提交 c4d6d90467af9ffa94772795d5c5d191228933c1
作者:格雷格·培根 <gbacon@dbresearch.net>
日期:2011 年 4 月 7 日星期四 12:20:53 -0500

    批判的

差异 --git a/关键 b/关键
新文件模式 100644
索引 0000000..82d423c
--- /dev/null
+++ b/关键
@@ -0,0 +1 @@
+非常重要的信息

甚至git-clone没有注意到!

$ 光盘 ..
$ git clone 回购克隆
克隆到克隆...
完毕。
$ 猫克隆/关键
非常重要的信息

执行完整性检查的特定 git 命令模式(例如git show $sha1应该存在但不存在git show)的完整列表是什么?git show HEAD

4

2 回答 2

6

为了回应 Mark Longair 的回答,我启动了 cscope 并发现:

注意 cscope 有一个 curses 接口并且很好地集成到 Vim 中,以防你引起你的兴趣

Functions calling this function: parse_object

  File              Function                       Line
0 bundle.c          verify_bundle                   110 struct object *o = parse_object(e->sha1);
1 bundle.c          create_bundle                   242 struct object *object = parse_object(sha1);
2 bundle.c          create_bundle                   247 struct object *object = parse_object(sha1);
3 bundle.c          create_bundle                   323 obj = parse_object(sha1);
4 commit.c          lookup_commit_reference_gently   30 struct object *obj = deref_tag(parse_object(sha1), NULL, 0);
5 http-backend.c    show_text_ref                   372 struct object *o = parse_object(sha1);
6 http-push.c       one_remote_object               742 obj = parse_object(sha1);
7 http-push.c       add_remote_info_ref            1530 o = parse_object(ref->old_sha1);
8 log-tree.c        add_ref_decoration               93 struct object *obj = parse_object(sha1);
9 merge-recursive.c get_ref                        1664 object = deref_tag(parse_object(sha1), name, strlen(name));
a pack-refs.c       handle_one_ref                   43 struct object *o = parse_object(sha1);
b pretty.c          format_commit_one               835 parse_object(commit->object.sha1);
c reachable.c       add_one_reflog_ent              122 object = parse_object(osha1);
d reachable.c       add_one_reflog_ent              125 object = parse_object(nsha1);
e reachable.c       add_one_ref                     133 struct object *object = parse_object(sha1);
f reflog-walk.c     fake_reflog_parent              234 commit_info->commit = (struct commit *)parse_object(reflog->osha1);
g refs.c            peel_ref                        647 o = parse_object(base);
h refs.c            write_ref_sha1                 1452 o = parse_object(sha1);
i remote.c          ref_newer                      1482 o = deref_tag(parse_object(old_sha1), NULL, 0);
j remote.c          ref_newer                      1487 o = deref_tag(parse_object(new_sha1), NULL, 0);
k revision.c        add_head_to_pending             166 obj = parse_object(sha1);
l revision.c        get_reference                   176 object = parse_object(sha1);
m revision.c        handle_commit                   196 object = parse_object(tag->tagged->sha1);
n revision.c        handle_one_reflog_commit        855 struct object *o = parse_object(sha1);
o server-info.c     add_info_ref                     12 struct object *o = parse_object(sha1);
p sha1_name.c       peel_to_type                    508 o = parse_object(sha1);
q sha1_name.c       peel_to_type                    511 if (!o || (!o->parsed && !parse_object(o->sha1)))
r sha1_name.c       peel_onion                      573 o = parse_object(outer);
s sha1_name.c       peel_onion                      578 if (!o || (!o->parsed && !parse_object(o->sha1)))
t sha1_name.c       handle_one_ref                  698 struct object *object = parse_object(sha1);
u sha1_name.c       get_sha1_oneline                740 if (!parse_object(commit->object.sha1))
v tag.c             deref_tag                        16 o = parse_object(((struct tag *)o)->tagged->sha1);
w tree.c            parse_tree_indirect             271 struct object *obj = parse_object(sha1);
x tree.c            parse_tree_indirect             284 parse_object(obj->sha1);
y upload-pack.c     got_sha1                        342 o = parse_object(sha1);
z upload-pack.c     reachable                       382 parse_object(commit->object.sha1);
A upload-pack.c     receive_needs                   526 object = parse_object(sha1);
B upload-pack.c     send_ref                        644 struct object *o = parse_object(sha1);
C upload-pack.c     mark_our_ref                    670 struct object *o = parse_object(sha1);
D walker.c          loop                            182 parse_object(obj->sha1);
于 2011-04-07T23:21:54.990 回答
6

以下是我将如何找出这一点,尽管我不会通过每个源文件来确定执行检查的条件。:)

克隆git的源代码:

git clone git://git.kernel.org/pub/scm/git/git.git

查看您关心的版本:

cd git
git checkout v1.7.1

查找该错误消息:

git grep 'sha1 mismatch'

这将导致您object.cparse_object功能。现在寻找那个函数:

git grep parse_object

...并通过 38 个文件检查调用该函数的条件。

于 2011-04-07T19:13:49.927 回答