| #define USE_THE_REPOSITORY_VARIABLE |
| |
| #include "builtin.h" |
| #include "environment.h" |
| #include "parse-options.h" |
| #include "path-walk.h" |
| #include "progress.h" |
| #include "quote.h" |
| #include "ref-filter.h" |
| #include "refs.h" |
| #include "revision.h" |
| #include "strbuf.h" |
| #include "string-list.h" |
| #include "shallow.h" |
| #include "utf8.h" |
| |
| static const char *const repo_usage[] = { |
| "git repo info [--format=(keyvalue|nul) | -z] [--all | <key>...]", |
| "git repo structure [--format=(table|keyvalue|nul) | -z]", |
| NULL |
| }; |
| |
| typedef int get_value_fn(struct repository *repo, struct strbuf *buf); |
| |
| enum output_format { |
| FORMAT_TABLE, |
| FORMAT_KEYVALUE, |
| FORMAT_NUL_TERMINATED, |
| }; |
| |
| struct field { |
| const char *key; |
| get_value_fn *get_value; |
| }; |
| |
| static int get_layout_bare(struct repository *repo UNUSED, struct strbuf *buf) |
| { |
| strbuf_addstr(buf, is_bare_repository() ? "true" : "false"); |
| return 0; |
| } |
| |
| static int get_layout_shallow(struct repository *repo, struct strbuf *buf) |
| { |
| strbuf_addstr(buf, |
| is_repository_shallow(repo) ? "true" : "false"); |
| return 0; |
| } |
| |
| static int get_object_format(struct repository *repo, struct strbuf *buf) |
| { |
| strbuf_addstr(buf, repo->hash_algo->name); |
| return 0; |
| } |
| |
| static int get_references_format(struct repository *repo, struct strbuf *buf) |
| { |
| strbuf_addstr(buf, |
| ref_storage_format_to_name(repo->ref_storage_format)); |
| return 0; |
| } |
| |
| /* repo_info_fields keys must be in lexicographical order */ |
| static const struct field repo_info_fields[] = { |
| { "layout.bare", get_layout_bare }, |
| { "layout.shallow", get_layout_shallow }, |
| { "object.format", get_object_format }, |
| { "references.format", get_references_format }, |
| }; |
| |
| static int repo_info_fields_cmp(const void *va, const void *vb) |
| { |
| const struct field *a = va; |
| const struct field *b = vb; |
| |
| return strcmp(a->key, b->key); |
| } |
| |
| static get_value_fn *get_value_fn_for_key(const char *key) |
| { |
| const struct field search_key = { key, NULL }; |
| const struct field *found = bsearch(&search_key, repo_info_fields, |
| ARRAY_SIZE(repo_info_fields), |
| sizeof(*found), |
| repo_info_fields_cmp); |
| return found ? found->get_value : NULL; |
| } |
| |
| static void print_field(enum output_format format, const char *key, |
| const char *value) |
| { |
| switch (format) { |
| case FORMAT_KEYVALUE: |
| printf("%s=", key); |
| quote_c_style(value, NULL, stdout, 0); |
| putchar('\n'); |
| break; |
| case FORMAT_NUL_TERMINATED: |
| printf("%s\n%s%c", key, value, '\0'); |
| break; |
| default: |
| BUG("not a valid output format: %d", format); |
| } |
| } |
| |
| static int print_fields(int argc, const char **argv, |
| struct repository *repo, |
| enum output_format format) |
| { |
| int ret = 0; |
| struct strbuf valbuf = STRBUF_INIT; |
| |
| for (int i = 0; i < argc; i++) { |
| get_value_fn *get_value; |
| const char *key = argv[i]; |
| |
| get_value = get_value_fn_for_key(key); |
| |
| if (!get_value) { |
| ret = error(_("key '%s' not found"), key); |
| continue; |
| } |
| |
| strbuf_reset(&valbuf); |
| get_value(repo, &valbuf); |
| print_field(format, key, valbuf.buf); |
| } |
| |
| strbuf_release(&valbuf); |
| return ret; |
| } |
| |
| static int print_all_fields(struct repository *repo, |
| enum output_format format) |
| { |
| struct strbuf valbuf = STRBUF_INIT; |
| |
| for (size_t i = 0; i < ARRAY_SIZE(repo_info_fields); i++) { |
| const struct field *field = &repo_info_fields[i]; |
| |
| strbuf_reset(&valbuf); |
| field->get_value(repo, &valbuf); |
| print_field(format, field->key, valbuf.buf); |
| } |
| |
| strbuf_release(&valbuf); |
| return 0; |
| } |
| |
| static int parse_format_cb(const struct option *opt, |
| const char *arg, int unset UNUSED) |
| { |
| enum output_format *format = opt->value; |
| |
| if (opt->short_name == 'z') |
| *format = FORMAT_NUL_TERMINATED; |
| else if (!strcmp(arg, "nul")) |
| *format = FORMAT_NUL_TERMINATED; |
| else if (!strcmp(arg, "keyvalue")) |
| *format = FORMAT_KEYVALUE; |
| else if (!strcmp(arg, "table")) |
| *format = FORMAT_TABLE; |
| else |
| die(_("invalid format '%s'"), arg); |
| |
| return 0; |
| } |
| |
| static int cmd_repo_info(int argc, const char **argv, const char *prefix, |
| struct repository *repo) |
| { |
| enum output_format format = FORMAT_KEYVALUE; |
| int all_keys = 0; |
| struct option options[] = { |
| OPT_CALLBACK_F(0, "format", &format, N_("format"), |
| N_("output format"), |
| PARSE_OPT_NONEG, parse_format_cb), |
| OPT_CALLBACK_F('z', NULL, &format, NULL, |
| N_("synonym for --format=nul"), |
| PARSE_OPT_NONEG | PARSE_OPT_NOARG, |
| parse_format_cb), |
| OPT_BOOL(0, "all", &all_keys, N_("print all keys/values")), |
| OPT_END() |
| }; |
| |
| argc = parse_options(argc, argv, prefix, options, repo_usage, 0); |
| if (format != FORMAT_KEYVALUE && format != FORMAT_NUL_TERMINATED) |
| die(_("unsupported output format")); |
| |
| if (all_keys && argc) |
| die(_("--all and <key> cannot be used together")); |
| |
| if (all_keys) |
| return print_all_fields(repo, format); |
| else |
| return print_fields(argc, argv, repo, format); |
| } |
| |
| struct ref_stats { |
| size_t branches; |
| size_t remotes; |
| size_t tags; |
| size_t others; |
| }; |
| |
| struct object_stats { |
| size_t tags; |
| size_t commits; |
| size_t trees; |
| size_t blobs; |
| }; |
| |
| struct repo_structure { |
| struct ref_stats refs; |
| struct object_stats objects; |
| }; |
| |
| struct stats_table { |
| struct string_list rows; |
| |
| int name_col_width; |
| int value_col_width; |
| }; |
| |
| /* |
| * Holds column data that gets stored for each row. |
| */ |
| struct stats_table_entry { |
| char *value; |
| }; |
| |
| static void stats_table_vaddf(struct stats_table *table, |
| struct stats_table_entry *entry, |
| const char *format, va_list ap) |
| { |
| struct strbuf buf = STRBUF_INIT; |
| struct string_list_item *item; |
| char *formatted_name; |
| int name_width; |
| |
| strbuf_vaddf(&buf, format, ap); |
| formatted_name = strbuf_detach(&buf, NULL); |
| name_width = utf8_strwidth(formatted_name); |
| |
| item = string_list_append_nodup(&table->rows, formatted_name); |
| item->util = entry; |
| |
| if (name_width > table->name_col_width) |
| table->name_col_width = name_width; |
| if (entry) { |
| int value_width = utf8_strwidth(entry->value); |
| if (value_width > table->value_col_width) |
| table->value_col_width = value_width; |
| } |
| } |
| |
| static void stats_table_addf(struct stats_table *table, const char *format, ...) |
| { |
| va_list ap; |
| |
| va_start(ap, format); |
| stats_table_vaddf(table, NULL, format, ap); |
| va_end(ap); |
| } |
| |
| static void stats_table_count_addf(struct stats_table *table, size_t value, |
| const char *format, ...) |
| { |
| struct stats_table_entry *entry; |
| va_list ap; |
| |
| CALLOC_ARRAY(entry, 1); |
| entry->value = xstrfmt("%" PRIuMAX, (uintmax_t)value); |
| |
| va_start(ap, format); |
| stats_table_vaddf(table, entry, format, ap); |
| va_end(ap); |
| } |
| |
| static inline size_t get_total_reference_count(struct ref_stats *stats) |
| { |
| return stats->branches + stats->remotes + stats->tags + stats->others; |
| } |
| |
| static inline size_t get_total_object_count(struct object_stats *stats) |
| { |
| return stats->tags + stats->commits + stats->trees + stats->blobs; |
| } |
| |
| static void stats_table_setup_structure(struct stats_table *table, |
| struct repo_structure *stats) |
| { |
| struct object_stats *objects = &stats->objects; |
| struct ref_stats *refs = &stats->refs; |
| size_t object_total; |
| size_t ref_total; |
| |
| ref_total = get_total_reference_count(refs); |
| stats_table_addf(table, "* %s", _("References")); |
| stats_table_count_addf(table, ref_total, " * %s", _("Count")); |
| stats_table_count_addf(table, refs->branches, " * %s", _("Branches")); |
| stats_table_count_addf(table, refs->tags, " * %s", _("Tags")); |
| stats_table_count_addf(table, refs->remotes, " * %s", _("Remotes")); |
| stats_table_count_addf(table, refs->others, " * %s", _("Others")); |
| |
| object_total = get_total_object_count(objects); |
| stats_table_addf(table, ""); |
| stats_table_addf(table, "* %s", _("Reachable objects")); |
| stats_table_count_addf(table, object_total, " * %s", _("Count")); |
| stats_table_count_addf(table, objects->commits, " * %s", _("Commits")); |
| stats_table_count_addf(table, objects->trees, " * %s", _("Trees")); |
| stats_table_count_addf(table, objects->blobs, " * %s", _("Blobs")); |
| stats_table_count_addf(table, objects->tags, " * %s", _("Tags")); |
| } |
| |
| static void stats_table_print_structure(const struct stats_table *table) |
| { |
| const char *name_col_title = _("Repository structure"); |
| const char *value_col_title = _("Value"); |
| int name_col_width = utf8_strwidth(name_col_title); |
| int value_col_width = utf8_strwidth(value_col_title); |
| struct string_list_item *item; |
| struct strbuf buf = STRBUF_INIT; |
| |
| if (table->name_col_width > name_col_width) |
| name_col_width = table->name_col_width; |
| if (table->value_col_width > value_col_width) |
| value_col_width = table->value_col_width; |
| |
| strbuf_addstr(&buf, "| "); |
| strbuf_utf8_align(&buf, ALIGN_LEFT, name_col_width, name_col_title); |
| strbuf_addstr(&buf, " | "); |
| strbuf_utf8_align(&buf, ALIGN_LEFT, value_col_width, value_col_title); |
| strbuf_addstr(&buf, " |"); |
| printf("%s\n", buf.buf); |
| |
| printf("| "); |
| for (int i = 0; i < name_col_width; i++) |
| putchar('-'); |
| printf(" | "); |
| for (int i = 0; i < value_col_width; i++) |
| putchar('-'); |
| printf(" |\n"); |
| |
| for_each_string_list_item(item, &table->rows) { |
| struct stats_table_entry *entry = item->util; |
| const char *value = ""; |
| |
| if (entry) { |
| struct stats_table_entry *entry = item->util; |
| value = entry->value; |
| } |
| |
| strbuf_reset(&buf); |
| strbuf_addstr(&buf, "| "); |
| strbuf_utf8_align(&buf, ALIGN_LEFT, name_col_width, item->string); |
| strbuf_addstr(&buf, " | "); |
| strbuf_utf8_align(&buf, ALIGN_RIGHT, value_col_width, value); |
| strbuf_addstr(&buf, " |"); |
| printf("%s\n", buf.buf); |
| } |
| |
| strbuf_release(&buf); |
| } |
| |
| static void stats_table_clear(struct stats_table *table) |
| { |
| struct stats_table_entry *entry; |
| struct string_list_item *item; |
| |
| for_each_string_list_item(item, &table->rows) { |
| entry = item->util; |
| if (entry) |
| free(entry->value); |
| } |
| |
| string_list_clear(&table->rows, 1); |
| } |
| |
| static void structure_keyvalue_print(struct repo_structure *stats, |
| char key_delim, char value_delim) |
| { |
| printf("references.branches.count%c%" PRIuMAX "%c", key_delim, |
| (uintmax_t)stats->refs.branches, value_delim); |
| printf("references.tags.count%c%" PRIuMAX "%c", key_delim, |
| (uintmax_t)stats->refs.tags, value_delim); |
| printf("references.remotes.count%c%" PRIuMAX "%c", key_delim, |
| (uintmax_t)stats->refs.remotes, value_delim); |
| printf("references.others.count%c%" PRIuMAX "%c", key_delim, |
| (uintmax_t)stats->refs.others, value_delim); |
| |
| printf("objects.commits.count%c%" PRIuMAX "%c", key_delim, |
| (uintmax_t)stats->objects.commits, value_delim); |
| printf("objects.trees.count%c%" PRIuMAX "%c", key_delim, |
| (uintmax_t)stats->objects.trees, value_delim); |
| printf("objects.blobs.count%c%" PRIuMAX "%c", key_delim, |
| (uintmax_t)stats->objects.blobs, value_delim); |
| printf("objects.tags.count%c%" PRIuMAX "%c", key_delim, |
| (uintmax_t)stats->objects.tags, value_delim); |
| |
| fflush(stdout); |
| } |
| |
| struct count_references_data { |
| struct ref_stats *stats; |
| struct rev_info *revs; |
| struct progress *progress; |
| }; |
| |
| static int count_references(const struct reference *ref, void *cb_data) |
| { |
| struct count_references_data *data = cb_data; |
| struct ref_stats *stats = data->stats; |
| size_t ref_count; |
| |
| switch (ref_kind_from_refname(ref->name)) { |
| case FILTER_REFS_BRANCHES: |
| stats->branches++; |
| break; |
| case FILTER_REFS_REMOTES: |
| stats->remotes++; |
| break; |
| case FILTER_REFS_TAGS: |
| stats->tags++; |
| break; |
| case FILTER_REFS_OTHERS: |
| stats->others++; |
| break; |
| default: |
| BUG("unexpected reference type"); |
| } |
| |
| /* |
| * While iterating through references for counting, also add OIDs in |
| * preparation for the path walk. |
| */ |
| add_pending_oid(data->revs, NULL, ref->oid, 0); |
| |
| ref_count = get_total_reference_count(stats); |
| display_progress(data->progress, ref_count); |
| |
| return 0; |
| } |
| |
| static void structure_count_references(struct ref_stats *stats, |
| struct rev_info *revs, |
| struct repository *repo, |
| int show_progress) |
| { |
| struct count_references_data data = { |
| .stats = stats, |
| .revs = revs, |
| }; |
| |
| if (show_progress) |
| data.progress = start_delayed_progress(repo, |
| _("Counting references"), 0); |
| |
| refs_for_each_ref(get_main_ref_store(repo), count_references, &data); |
| stop_progress(&data.progress); |
| } |
| |
| struct count_objects_data { |
| struct object_stats *stats; |
| struct progress *progress; |
| }; |
| |
| static int count_objects(const char *path UNUSED, struct oid_array *oids, |
| enum object_type type, void *cb_data) |
| { |
| struct count_objects_data *data = cb_data; |
| struct object_stats *stats = data->stats; |
| size_t object_count; |
| |
| switch (type) { |
| case OBJ_TAG: |
| stats->tags += oids->nr; |
| break; |
| case OBJ_COMMIT: |
| stats->commits += oids->nr; |
| break; |
| case OBJ_TREE: |
| stats->trees += oids->nr; |
| break; |
| case OBJ_BLOB: |
| stats->blobs += oids->nr; |
| break; |
| default: |
| BUG("invalid object type"); |
| } |
| |
| object_count = get_total_object_count(stats); |
| display_progress(data->progress, object_count); |
| |
| return 0; |
| } |
| |
| static void structure_count_objects(struct object_stats *stats, |
| struct rev_info *revs, |
| struct repository *repo, int show_progress) |
| { |
| struct path_walk_info info = PATH_WALK_INFO_INIT; |
| struct count_objects_data data = { |
| .stats = stats, |
| }; |
| |
| info.revs = revs; |
| info.path_fn = count_objects; |
| info.path_fn_data = &data; |
| |
| if (show_progress) |
| data.progress = start_delayed_progress(repo, _("Counting objects"), 0); |
| |
| walk_objects_by_path(&info); |
| path_walk_info_clear(&info); |
| stop_progress(&data.progress); |
| } |
| |
| static int cmd_repo_structure(int argc, const char **argv, const char *prefix, |
| struct repository *repo) |
| { |
| struct stats_table table = { |
| .rows = STRING_LIST_INIT_DUP, |
| }; |
| enum output_format format = FORMAT_TABLE; |
| struct repo_structure stats = { 0 }; |
| struct rev_info revs; |
| int show_progress = -1; |
| struct option options[] = { |
| OPT_CALLBACK_F(0, "format", &format, N_("format"), |
| N_("output format"), |
| PARSE_OPT_NONEG, parse_format_cb), |
| OPT_CALLBACK_F('z', NULL, &format, NULL, |
| N_("synonym for --format=nul"), |
| PARSE_OPT_NONEG | PARSE_OPT_NOARG, |
| parse_format_cb), |
| OPT_BOOL(0, "progress", &show_progress, N_("show progress")), |
| OPT_END() |
| }; |
| |
| argc = parse_options(argc, argv, prefix, options, repo_usage, 0); |
| if (argc) |
| usage(_("too many arguments")); |
| |
| repo_init_revisions(repo, &revs, prefix); |
| |
| if (show_progress < 0) |
| show_progress = isatty(2); |
| |
| structure_count_references(&stats.refs, &revs, repo, show_progress); |
| structure_count_objects(&stats.objects, &revs, repo, show_progress); |
| |
| switch (format) { |
| case FORMAT_TABLE: |
| stats_table_setup_structure(&table, &stats); |
| stats_table_print_structure(&table); |
| break; |
| case FORMAT_KEYVALUE: |
| structure_keyvalue_print(&stats, '=', '\n'); |
| break; |
| case FORMAT_NUL_TERMINATED: |
| structure_keyvalue_print(&stats, '\n', '\0'); |
| break; |
| default: |
| BUG("invalid output format"); |
| } |
| |
| stats_table_clear(&table); |
| release_revisions(&revs); |
| |
| return 0; |
| } |
| |
| int cmd_repo(int argc, const char **argv, const char *prefix, |
| struct repository *repo) |
| { |
| parse_opt_subcommand_fn *fn = NULL; |
| struct option options[] = { |
| OPT_SUBCOMMAND("info", &fn, cmd_repo_info), |
| OPT_SUBCOMMAND("structure", &fn, cmd_repo_structure), |
| OPT_END() |
| }; |
| |
| argc = parse_options(argc, argv, prefix, options, repo_usage, 0); |
| |
| return fn(argc, argv, prefix, repo); |
| } |