6c0b5b17e00cde0f42df54373757abb8269db666
   1#include "builtin.h"
   2#include "config.h"
   3#include "dir.h"
   4#include "lockfile.h"
   5#include "parse-options.h"
   6#include "repository.h"
   7#include "commit-graph.h"
   8
   9static char const * const builtin_commit_graph_usage[] = {
  10        N_("git commit-graph [--object-dir <objdir>]"),
  11        N_("git commit-graph read [--object-dir <objdir>]"),
  12        N_("git commit-graph verify [--object-dir <objdir>]"),
  13        N_("git commit-graph write [--object-dir <objdir>] [--append|--split] [--reachable|--stdin-packs|--stdin-commits] <split options>"),
  14        NULL
  15};
  16
  17static const char * const builtin_commit_graph_verify_usage[] = {
  18        N_("git commit-graph verify [--object-dir <objdir>]"),
  19        NULL
  20};
  21
  22static const char * const builtin_commit_graph_read_usage[] = {
  23        N_("git commit-graph read [--object-dir <objdir>]"),
  24        NULL
  25};
  26
  27static const char * const builtin_commit_graph_write_usage[] = {
  28        N_("git commit-graph write [--object-dir <objdir>] [--append|--split] [--reachable|--stdin-packs|--stdin-commits] <split options>"),
  29        NULL
  30};
  31
  32static struct opts_commit_graph {
  33        const char *obj_dir;
  34        int reachable;
  35        int stdin_packs;
  36        int stdin_commits;
  37        int append;
  38        int split;
  39} opts;
  40
  41static int graph_verify(int argc, const char **argv)
  42{
  43        struct commit_graph *graph = NULL;
  44        char *graph_name;
  45        int open_ok;
  46        int fd;
  47        struct stat st;
  48
  49        static struct option builtin_commit_graph_verify_options[] = {
  50                OPT_STRING(0, "object-dir", &opts.obj_dir,
  51                           N_("dir"),
  52                           N_("The object directory to store the graph")),
  53                OPT_END(),
  54        };
  55
  56        argc = parse_options(argc, argv, NULL,
  57                             builtin_commit_graph_verify_options,
  58                             builtin_commit_graph_verify_usage, 0);
  59
  60        if (!opts.obj_dir)
  61                opts.obj_dir = get_object_directory();
  62
  63        graph_name = get_commit_graph_filename(opts.obj_dir);
  64        open_ok = open_commit_graph(graph_name, &fd, &st);
  65        if (!open_ok && errno == ENOENT)
  66                return 0;
  67        if (!open_ok)
  68                die_errno(_("Could not open commit-graph '%s'"), graph_name);
  69        graph = load_commit_graph_one_fd_st(fd, &st);
  70        FREE_AND_NULL(graph_name);
  71
  72        if (!graph)
  73                return 1;
  74
  75        UNLEAK(graph);
  76        return verify_commit_graph(the_repository, graph);
  77}
  78
  79static int graph_read(int argc, const char **argv)
  80{
  81        struct commit_graph *graph = NULL;
  82        char *graph_name;
  83        int open_ok;
  84        int fd;
  85        struct stat st;
  86
  87        static struct option builtin_commit_graph_read_options[] = {
  88                OPT_STRING(0, "object-dir", &opts.obj_dir,
  89                        N_("dir"),
  90                        N_("The object directory to store the graph")),
  91                OPT_END(),
  92        };
  93
  94        argc = parse_options(argc, argv, NULL,
  95                             builtin_commit_graph_read_options,
  96                             builtin_commit_graph_read_usage, 0);
  97
  98        if (!opts.obj_dir)
  99                opts.obj_dir = get_object_directory();
 100
 101        graph_name = get_commit_graph_filename(opts.obj_dir);
 102
 103        open_ok = open_commit_graph(graph_name, &fd, &st);
 104        if (!open_ok)
 105                die_errno(_("Could not open commit-graph '%s'"), graph_name);
 106
 107        graph = load_commit_graph_one_fd_st(fd, &st);
 108        if (!graph)
 109                return 1;
 110
 111        FREE_AND_NULL(graph_name);
 112
 113        printf("header: %08x %d %d %d %d\n",
 114                ntohl(*(uint32_t*)graph->data),
 115                *(unsigned char*)(graph->data + 4),
 116                *(unsigned char*)(graph->data + 5),
 117                *(unsigned char*)(graph->data + 6),
 118                *(unsigned char*)(graph->data + 7));
 119        printf("num_commits: %u\n", graph->num_commits);
 120        printf("chunks:");
 121
 122        if (graph->chunk_oid_fanout)
 123                printf(" oid_fanout");
 124        if (graph->chunk_oid_lookup)
 125                printf(" oid_lookup");
 126        if (graph->chunk_commit_data)
 127                printf(" commit_metadata");
 128        if (graph->chunk_extra_edges)
 129                printf(" extra_edges");
 130        printf("\n");
 131
 132        UNLEAK(graph);
 133
 134        return 0;
 135}
 136
 137extern int read_replace_refs;
 138static struct split_commit_graph_opts split_opts;
 139
 140static int graph_write(int argc, const char **argv)
 141{
 142        struct string_list *pack_indexes = NULL;
 143        struct string_list *commit_hex = NULL;
 144        struct string_list lines;
 145        int result = 0;
 146        unsigned int flags = COMMIT_GRAPH_PROGRESS;
 147
 148        static struct option builtin_commit_graph_write_options[] = {
 149                OPT_STRING(0, "object-dir", &opts.obj_dir,
 150                        N_("dir"),
 151                        N_("The object directory to store the graph")),
 152                OPT_BOOL(0, "reachable", &opts.reachable,
 153                        N_("start walk at all refs")),
 154                OPT_BOOL(0, "stdin-packs", &opts.stdin_packs,
 155                        N_("scan pack-indexes listed by stdin for commits")),
 156                OPT_BOOL(0, "stdin-commits", &opts.stdin_commits,
 157                        N_("start walk at commits listed by stdin")),
 158                OPT_BOOL(0, "append", &opts.append,
 159                        N_("include all commits already in the commit-graph file")),
 160                OPT_BOOL(0, "split", &opts.split,
 161                        N_("allow writing an incremental commit-graph file")),
 162                OPT_INTEGER(0, "max-commits", &split_opts.max_commits,
 163                        N_("maximum number of commits in a non-base split commit-graph")),
 164                OPT_INTEGER(0, "size-multiple", &split_opts.size_multiple,
 165                        N_("maximum ratio between two levels of a split commit-graph")),
 166                OPT_EXPIRY_DATE(0, "expire-time", &split_opts.expire_time,
 167                        N_("maximum number of commits in a non-base split commit-graph")),
 168                OPT_END(),
 169        };
 170
 171        split_opts.size_multiple = 2;
 172        split_opts.max_commits = 0;
 173        split_opts.expire_time = 0;
 174
 175        argc = parse_options(argc, argv, NULL,
 176                             builtin_commit_graph_write_options,
 177                             builtin_commit_graph_write_usage, 0);
 178
 179        if (opts.reachable + opts.stdin_packs + opts.stdin_commits > 1)
 180                die(_("use at most one of --reachable, --stdin-commits, or --stdin-packs"));
 181        if (!opts.obj_dir)
 182                opts.obj_dir = get_object_directory();
 183        if (opts.append)
 184                flags |= COMMIT_GRAPH_APPEND;
 185        if (opts.split)
 186                flags |= COMMIT_GRAPH_SPLIT;
 187
 188        read_replace_refs = 0;
 189
 190        if (opts.reachable) {
 191                if (write_commit_graph_reachable(opts.obj_dir, flags, &split_opts))
 192                        return 1;
 193                return 0;
 194        }
 195
 196        string_list_init(&lines, 0);
 197        if (opts.stdin_packs || opts.stdin_commits) {
 198                struct strbuf buf = STRBUF_INIT;
 199
 200                while (strbuf_getline(&buf, stdin) != EOF)
 201                        string_list_append(&lines, strbuf_detach(&buf, NULL));
 202
 203                if (opts.stdin_packs)
 204                        pack_indexes = &lines;
 205                if (opts.stdin_commits)
 206                        commit_hex = &lines;
 207
 208                UNLEAK(buf);
 209        }
 210
 211        if (write_commit_graph(opts.obj_dir,
 212                               pack_indexes,
 213                               commit_hex,
 214                               flags,
 215                               &split_opts))
 216                result = 1;
 217
 218        UNLEAK(lines);
 219        return result;
 220}
 221
 222int cmd_commit_graph(int argc, const char **argv, const char *prefix)
 223{
 224        static struct option builtin_commit_graph_options[] = {
 225                OPT_STRING(0, "object-dir", &opts.obj_dir,
 226                        N_("dir"),
 227                        N_("The object directory to store the graph")),
 228                OPT_END(),
 229        };
 230
 231        if (argc == 2 && !strcmp(argv[1], "-h"))
 232                usage_with_options(builtin_commit_graph_usage,
 233                                   builtin_commit_graph_options);
 234
 235        git_config(git_default_config, NULL);
 236        argc = parse_options(argc, argv, prefix,
 237                             builtin_commit_graph_options,
 238                             builtin_commit_graph_usage,
 239                             PARSE_OPT_STOP_AT_NON_OPTION);
 240
 241        if (argc > 0) {
 242                if (!strcmp(argv[0], "read"))
 243                        return graph_read(argc, argv);
 244                if (!strcmp(argv[0], "verify"))
 245                        return graph_verify(argc, argv);
 246                if (!strcmp(argv[0], "write"))
 247                        return graph_write(argc, argv);
 248        }
 249
 250        usage_with_options(builtin_commit_graph_usage,
 251                           builtin_commit_graph_options);
 252}