268 lines
6.1 KiB
C
268 lines
6.1 KiB
C
#include "git-compat-util.h"
|
|
#include "hash.h"
|
|
#include "hex.h"
|
|
#include "reftable/system.h"
|
|
#include "reftable/reftable-constants.h"
|
|
#include "reftable/reftable-error.h"
|
|
#include "reftable/reftable-merged.h"
|
|
#include "reftable/reftable-stack.h"
|
|
#include "reftable/reftable-table.h"
|
|
#include "test-tool.h"
|
|
|
|
static void print_help(void)
|
|
{
|
|
printf("usage: dump [-st] arg\n\n"
|
|
"options: \n"
|
|
" -b dump blocks\n"
|
|
" -t dump table\n"
|
|
" -s dump stack\n"
|
|
" -6 sha256 hash format\n"
|
|
" -h this help\n"
|
|
"\n");
|
|
}
|
|
|
|
static int dump_blocks(const char *tablename)
|
|
{
|
|
struct reftable_table_iterator ti = { 0 };
|
|
struct reftable_block_source src = { 0 };
|
|
struct reftable_table *table = NULL;
|
|
uint8_t section_type = 0;
|
|
int err;
|
|
|
|
err = reftable_block_source_from_file(&src, tablename);
|
|
if (err < 0)
|
|
goto done;
|
|
|
|
err = reftable_table_new(&table, &src, tablename);
|
|
if (err < 0)
|
|
goto done;
|
|
|
|
err = reftable_table_iterator_init(&ti, table);
|
|
if (err < 0)
|
|
goto done;
|
|
|
|
printf("header:\n");
|
|
printf(" block_size: %d\n", table->block_size);
|
|
|
|
while (1) {
|
|
const struct reftable_block *block;
|
|
|
|
err = reftable_table_iterator_next(&ti, &block);
|
|
if (err < 0)
|
|
goto done;
|
|
if (err > 0)
|
|
break;
|
|
|
|
if (block->block_type != section_type) {
|
|
const char *section;
|
|
switch (block->block_type) {
|
|
case REFTABLE_BLOCK_TYPE_LOG:
|
|
section = "log";
|
|
break;
|
|
case REFTABLE_BLOCK_TYPE_REF:
|
|
section = "ref";
|
|
break;
|
|
case REFTABLE_BLOCK_TYPE_OBJ:
|
|
section = "obj";
|
|
break;
|
|
case REFTABLE_BLOCK_TYPE_INDEX:
|
|
section = "idx";
|
|
break;
|
|
default:
|
|
err = -1;
|
|
goto done;
|
|
}
|
|
|
|
section_type = block->block_type;
|
|
printf("%s:\n", section);
|
|
}
|
|
|
|
printf(" - length: %u\n", block->restart_off);
|
|
printf(" restarts: %u\n", block->restart_count);
|
|
}
|
|
|
|
done:
|
|
reftable_table_iterator_release(&ti);
|
|
reftable_table_decref(table);
|
|
return err;
|
|
}
|
|
|
|
static int dump_table(struct reftable_merged_table *mt)
|
|
{
|
|
struct reftable_iterator it = { NULL };
|
|
struct reftable_ref_record ref = { NULL };
|
|
struct reftable_log_record log = { NULL };
|
|
const struct git_hash_algo *algop;
|
|
int err;
|
|
|
|
err = reftable_merged_table_init_ref_iterator(mt, &it);
|
|
if (err < 0)
|
|
return err;
|
|
|
|
err = reftable_iterator_seek_ref(&it, "");
|
|
if (err < 0)
|
|
return err;
|
|
|
|
algop = &hash_algos[hash_algo_by_id(reftable_merged_table_hash_id(mt))];
|
|
|
|
while (1) {
|
|
err = reftable_iterator_next_ref(&it, &ref);
|
|
if (err > 0)
|
|
break;
|
|
if (err < 0)
|
|
return err;
|
|
|
|
printf("ref{%s(%" PRIu64 ") ", ref.refname, ref.update_index);
|
|
switch (ref.value_type) {
|
|
case REFTABLE_REF_SYMREF:
|
|
printf("=> %s", ref.value.symref);
|
|
break;
|
|
case REFTABLE_REF_VAL2:
|
|
printf("val 2 %s", hash_to_hex_algop(ref.value.val2.value, algop));
|
|
printf("(T %s)", hash_to_hex_algop(ref.value.val2.target_value, algop));
|
|
break;
|
|
case REFTABLE_REF_VAL1:
|
|
printf("val 1 %s", hash_to_hex_algop(ref.value.val1, algop));
|
|
break;
|
|
case REFTABLE_REF_DELETION:
|
|
printf("delete");
|
|
break;
|
|
}
|
|
printf("}\n");
|
|
}
|
|
reftable_iterator_destroy(&it);
|
|
reftable_ref_record_release(&ref);
|
|
|
|
err = reftable_merged_table_init_log_iterator(mt, &it);
|
|
if (err < 0)
|
|
return err;
|
|
|
|
err = reftable_iterator_seek_log(&it, "");
|
|
if (err < 0)
|
|
return err;
|
|
|
|
while (1) {
|
|
err = reftable_iterator_next_log(&it, &log);
|
|
if (err > 0)
|
|
break;
|
|
if (err < 0)
|
|
return err;
|
|
|
|
switch (log.value_type) {
|
|
case REFTABLE_LOG_DELETION:
|
|
printf("log{%s(%" PRIu64 ") delete\n", log.refname,
|
|
log.update_index);
|
|
break;
|
|
case REFTABLE_LOG_UPDATE:
|
|
printf("log{%s(%" PRIu64 ") %s <%s> %" PRIu64 " %04d\n",
|
|
log.refname, log.update_index,
|
|
log.value.update.name ? log.value.update.name : "",
|
|
log.value.update.email ? log.value.update.email : "",
|
|
log.value.update.time,
|
|
log.value.update.tz_offset);
|
|
printf("%s => ", hash_to_hex_algop(log.value.update.old_hash, algop));
|
|
printf("%s\n\n%s\n}\n", hash_to_hex_algop(log.value.update.new_hash, algop),
|
|
log.value.update.message ? log.value.update.message : "");
|
|
break;
|
|
}
|
|
}
|
|
reftable_iterator_destroy(&it);
|
|
reftable_log_record_release(&log);
|
|
return 0;
|
|
}
|
|
|
|
static int dump_stack(const char *stackdir, uint32_t hash_id)
|
|
{
|
|
struct reftable_stack *stack = NULL;
|
|
struct reftable_write_options opts = { .hash_id = hash_id };
|
|
struct reftable_merged_table *merged = NULL;
|
|
|
|
int err = reftable_new_stack(&stack, stackdir, &opts);
|
|
if (err < 0)
|
|
goto done;
|
|
|
|
merged = reftable_stack_merged_table(stack);
|
|
err = dump_table(merged);
|
|
done:
|
|
if (stack)
|
|
reftable_stack_destroy(stack);
|
|
return err;
|
|
}
|
|
|
|
static int dump_reftable(const char *tablename)
|
|
{
|
|
struct reftable_block_source src = { 0 };
|
|
struct reftable_merged_table *mt = NULL;
|
|
struct reftable_table *table = NULL;
|
|
int err;
|
|
|
|
err = reftable_block_source_from_file(&src, tablename);
|
|
if (err < 0)
|
|
goto done;
|
|
|
|
err = reftable_table_new(&table, &src, tablename);
|
|
if (err < 0)
|
|
goto done;
|
|
|
|
err = reftable_merged_table_new(&mt, &table, 1,
|
|
reftable_table_hash_id(table));
|
|
if (err < 0)
|
|
goto done;
|
|
|
|
err = dump_table(mt);
|
|
|
|
done:
|
|
reftable_merged_table_free(mt);
|
|
reftable_table_decref(table);
|
|
return err;
|
|
}
|
|
|
|
int cmd__dump_reftable(int argc, const char **argv)
|
|
{
|
|
int err = 0;
|
|
int opt_dump_blocks = 0;
|
|
int opt_dump_table = 0;
|
|
int opt_dump_stack = 0;
|
|
uint32_t opt_hash_id = REFTABLE_HASH_SHA1;
|
|
const char *arg = NULL, *argv0 = argv[0];
|
|
|
|
for (; argc > 1; argv++, argc--)
|
|
if (*argv[1] != '-')
|
|
break;
|
|
else if (!strcmp("-b", argv[1]))
|
|
opt_dump_blocks = 1;
|
|
else if (!strcmp("-t", argv[1]))
|
|
opt_dump_table = 1;
|
|
else if (!strcmp("-6", argv[1]))
|
|
opt_hash_id = REFTABLE_HASH_SHA256;
|
|
else if (!strcmp("-s", argv[1]))
|
|
opt_dump_stack = 1;
|
|
else if (!strcmp("-?", argv[1]) || !strcmp("-h", argv[1])) {
|
|
print_help();
|
|
return 2;
|
|
}
|
|
|
|
if (argc != 2) {
|
|
fprintf(stderr, "need argument\n");
|
|
print_help();
|
|
return 2;
|
|
}
|
|
|
|
arg = argv[1];
|
|
|
|
if (opt_dump_blocks) {
|
|
err = dump_blocks(arg);
|
|
} else if (opt_dump_table) {
|
|
err = dump_reftable(arg);
|
|
} else if (opt_dump_stack) {
|
|
err = dump_stack(arg, opt_hash_id);
|
|
}
|
|
|
|
if (err < 0) {
|
|
fprintf(stderr, "%s: %s: %s\n", argv0, arg,
|
|
reftable_error_str(err));
|
|
return 1;
|
|
}
|
|
return 0;
|
|
}
|