While running 'git commit-graph check', verify that the object IDs
are listed in lexicographic order and that the fanout table correctly
navigates into that list of object IDs.

In anticipation of checking the commits in the commit-graph file
against the object database, parse the commits from that file in
advance. We perform this parse now to ensure the object cache contains
only commits from this commit-graph file.

Signed-off-by: Derrick Stolee <dsto...@microsoft.com>
---
 commit-graph.c | 34 ++++++++++++++++++++++++++++++++++
 1 file changed, 34 insertions(+)

diff --git a/commit-graph.c b/commit-graph.c
index 6d0d303a7a..6e3c08cd5c 100644
--- a/commit-graph.c
+++ b/commit-graph.c
@@ -835,6 +835,9 @@ static int check_commit_graph_error;
 
 int check_commit_graph(struct commit_graph *g)
 {
+       uint32_t i, cur_fanout_pos = 0;
+       struct object_id prev_oid, cur_oid;
+
        if (!g) {
                graph_report(_("no commit-graph file loaded"));
                return 1;
@@ -859,5 +862,36 @@ int check_commit_graph(struct commit_graph *g)
        if (g->hash_len != GRAPH_OID_LEN)
                graph_report(_("commit-graph has incorrect hash length: %d"), 
g->hash_len);
 
+       for (i = 0; i < g->num_commits; i++) {
+               struct commit *graph_commit;
+
+               hashcpy(cur_oid.hash, g->chunk_oid_lookup + g->hash_len * i);
+
+               if (i > 0 && oidcmp(&prev_oid, &cur_oid) >= 0)
+                       graph_report(_("commit-graph has incorrect oid order: 
%s then %s"),
+
+               oid_to_hex(&prev_oid),
+               oid_to_hex(&cur_oid));
+               oidcpy(&prev_oid, &cur_oid);
+
+               while (cur_oid.hash[0] > cur_fanout_pos) {
+                       uint32_t fanout_value = get_be32(g->chunk_oid_fanout + 
cur_fanout_pos);
+                       if (i != fanout_value)
+                               graph_report(_("commit-graph has incorrect 
fanout value: fanout[%d] = %u != %u"),
+                                            cur_fanout_pos, fanout_value, i);
+
+                       cur_fanout_pos++;
+               }
+
+               graph_commit = lookup_commit(&cur_oid);
+
+               if (!parse_commit_in_graph_one(g, graph_commit))
+                       graph_report(_("failed to parse %s from commit-graph"), 
oid_to_hex(&cur_oid));
+
+               if (graph_commit->graph_pos != i)
+                       graph_report(_("graph_pos for commit %s is %u != %u"), 
oid_to_hex(&cur_oid),
+                                    graph_commit->graph_pos, i);
+       }
+
        return check_commit_graph_error;
 }
-- 
2.17.0.39.g685157f7fb

Reply via email to