Merge branch 'jk/repo-info-path-keys' into seen

The "git repo info" command has been taught new keys to output both
absolute and relative paths for "gitdir" and "commondir", supported by
a new path-formatting helper extracted from "git rev-parse".

* jk/repo-info-path-keys:
  repo: add path.gitdir with absolute and relative suffix formatting
  repo: add path.commondir with absolute and relative suffix formatting
  rev-parse: use append_formatted_path() for path formatting
  path: introduce append_formatted_path() for shared path formatting
This commit is contained in:
Junio C Hamano
2026-06-15 10:27:29 -07:00
6 changed files with 262 additions and 80 deletions

View File

@@ -104,6 +104,21 @@ values that they return:
`object.format`::
The object format (hash algorithm) used in the repository.
`path.commondir.absolute`::
The canonical absolute path to the Git repository's common
directory (the shared `.git` directory containing objects,
refs, and global configuration).
`path.commondir.relative`::
The path to the Git repository's common directory relative to
the current working directory.
`path.gitdir.absolute`::
The canonical absolute path to the Git repository directory (the `.git` directory).
`path.gitdir.relative`::
The path to the Git repository directory relative to the current working directory.
`references.format`::
The reference storage format. The valid values are:
+

View File

@@ -7,12 +7,14 @@
#include "hex.h"
#include "odb.h"
#include "parse-options.h"
#include "path.h"
#include "path-walk.h"
#include "progress.h"
#include "quote.h"
#include "ref-filter.h"
#include "refs.h"
#include "revision.h"
#include "setup.h"
#include "strbuf.h"
#include "string-list.h"
#include "shallow.h"
@@ -75,6 +77,50 @@ static int get_object_format(struct repository *repo, struct strbuf *buf)
return 0;
}
static int get_path_commondir_absolute(struct repository *repo, struct strbuf *buf)
{
const char *common_dir = repo_get_common_dir(repo);
if (!common_dir)
return error(_("unable to get common directory"));
append_formatted_path(buf, common_dir, startup_info->prefix, PATH_FORMAT_CANONICAL);
return 0;
}
static int get_path_commondir_relative(struct repository *repo, struct strbuf *buf)
{
const char *common_dir = repo_get_common_dir(repo);
if (!common_dir)
return error(_("unable to get common directory"));
append_formatted_path(buf, common_dir, startup_info->prefix, PATH_FORMAT_RELATIVE);
return 0;
}
static int get_path_gitdir_absolute(struct repository *repo, struct strbuf *buf)
{
const char *git_dir = repo_get_git_dir(repo);
if (!git_dir)
return error(_("unable to get git directory"));
append_formatted_path(buf, git_dir, startup_info->prefix, PATH_FORMAT_CANONICAL);
return 0;
}
static int get_path_gitdir_relative(struct repository *repo, struct strbuf *buf)
{
const char *git_dir = repo_get_git_dir(repo);
if (!git_dir)
return error(_("unable to get git directory"));
append_formatted_path(buf, git_dir, startup_info->prefix, PATH_FORMAT_RELATIVE);
return 0;
}
static int get_references_format(struct repository *repo, struct strbuf *buf)
{
strbuf_addstr(buf,
@@ -87,6 +133,10 @@ static const struct repo_info_field repo_info_field[] = {
{ "layout.bare", get_layout_bare },
{ "layout.shallow", get_layout_shallow },
{ "object.format", get_object_format },
{ "path.commondir.absolute", get_path_commondir_absolute },
{ "path.commondir.relative", get_path_commondir_relative },
{ "path.gitdir.absolute", get_path_gitdir_absolute },
{ "path.gitdir.relative", get_path_gitdir_relative },
{ "references.format", get_references_format },
};

View File

@@ -633,73 +633,16 @@ static void handle_ref_opt(const char *pattern, const char *prefix)
clear_ref_exclusions(&ref_excludes);
}
enum format_type {
/* We would like a relative path. */
FORMAT_RELATIVE,
/* We would like a canonical absolute path. */
FORMAT_CANONICAL,
/* We would like the default behavior. */
FORMAT_DEFAULT,
};
enum default_type {
/* Our default is a relative path. */
DEFAULT_RELATIVE,
/* Our default is a relative path if there's a shared root. */
DEFAULT_RELATIVE_IF_SHARED,
/* Our default is a canonical absolute path. */
DEFAULT_CANONICAL,
/* Our default is not to modify the item. */
DEFAULT_UNMODIFIED,
};
static void print_path(const char *path, const char *prefix, enum format_type format, enum default_type def)
static void print_path(const char *path, const char *prefix,
enum path_format arg_path_format, enum path_format def_format)
{
char *cwd = NULL;
/*
* We don't ever produce a relative path if prefix is NULL, so set the
* prefix to the current directory so that we can produce a relative
* path whenever possible. If we're using RELATIVE_IF_SHARED mode, then
* we want an absolute path unless the two share a common prefix, so don't
* set it in that case, since doing so causes a relative path to always
* be produced if possible.
*/
if (!prefix && (format != FORMAT_DEFAULT || def != DEFAULT_RELATIVE_IF_SHARED))
prefix = cwd = xgetcwd();
if (format == FORMAT_DEFAULT && def == DEFAULT_UNMODIFIED) {
puts(path);
} else if (format == FORMAT_RELATIVE ||
(format == FORMAT_DEFAULT && def == DEFAULT_RELATIVE)) {
/*
* In order for relative_path to work as expected, we need to
* make sure that both paths are absolute paths. If we don't,
* we can end up with an unexpected absolute path that the user
* didn't want.
*/
struct strbuf buf = STRBUF_INIT, realbuf = STRBUF_INIT, prefixbuf = STRBUF_INIT;
if (!is_absolute_path(path)) {
strbuf_realpath_forgiving(&realbuf, path, 1);
path = realbuf.buf;
}
if (!is_absolute_path(prefix)) {
strbuf_realpath_forgiving(&prefixbuf, prefix, 1);
prefix = prefixbuf.buf;
}
puts(relative_path(path, prefix, &buf));
strbuf_release(&buf);
strbuf_release(&realbuf);
strbuf_release(&prefixbuf);
} else if (format == FORMAT_DEFAULT && def == DEFAULT_RELATIVE_IF_SHARED) {
struct strbuf buf = STRBUF_INIT;
puts(relative_path(path, prefix, &buf));
strbuf_release(&buf);
} else {
struct strbuf buf = STRBUF_INIT;
strbuf_realpath_forgiving(&buf, path, 1);
puts(buf.buf);
strbuf_release(&buf);
}
free(cwd);
struct strbuf sb = STRBUF_INIT;
enum path_format fmt = (arg_path_format != PATH_FORMAT_DEFAULT) ? arg_path_format : def_format;
append_formatted_path(&sb, path, prefix, fmt);
puts(sb.buf);
strbuf_release(&sb);
}
int cmd_rev_parse(int argc,
@@ -718,7 +661,7 @@ int cmd_rev_parse(int argc,
const char *name = NULL;
struct strbuf buf = STRBUF_INIT;
int seen_end_of_options = 0;
enum format_type format = FORMAT_DEFAULT;
enum path_format arg_path_format = PATH_FORMAT_DEFAULT;
show_usage_if_asked(argc, argv, builtin_rev_parse_usage);
@@ -798,8 +741,8 @@ int cmd_rev_parse(int argc,
die(_("--git-path requires an argument"));
print_path(repo_git_path_replace(the_repository, &buf,
"%s", argv[i + 1]), prefix,
format,
DEFAULT_RELATIVE_IF_SHARED);
arg_path_format,
PATH_FORMAT_RELATIVE_IF_SHARED);
i++;
continue;
}
@@ -821,9 +764,9 @@ int cmd_rev_parse(int argc,
if (!arg)
die(_("--path-format requires an argument"));
if (!strcmp(arg, "absolute")) {
format = FORMAT_CANONICAL;
arg_path_format = PATH_FORMAT_CANONICAL;
} else if (!strcmp(arg, "relative")) {
format = FORMAT_RELATIVE;
arg_path_format = PATH_FORMAT_RELATIVE;
} else {
die(_("unknown argument to --path-format: %s"), arg);
}
@@ -996,7 +939,7 @@ int cmd_rev_parse(int argc,
if (!strcmp(arg, "--show-toplevel")) {
const char *work_tree = repo_get_work_tree(the_repository);
if (work_tree)
print_path(work_tree, prefix, format, DEFAULT_UNMODIFIED);
print_path(work_tree, prefix, arg_path_format, PATH_FORMAT_UNMODIFIED);
else
die(_("this operation must be run in a work tree"));
continue;
@@ -1004,7 +947,7 @@ int cmd_rev_parse(int argc,
if (!strcmp(arg, "--show-superproject-working-tree")) {
struct strbuf superproject = STRBUF_INIT;
if (get_superproject_working_tree(&superproject))
print_path(superproject.buf, prefix, format, DEFAULT_UNMODIFIED);
print_path(superproject.buf, prefix, arg_path_format, PATH_FORMAT_UNMODIFIED);
strbuf_release(&superproject);
continue;
}
@@ -1039,18 +982,18 @@ int cmd_rev_parse(int argc,
const char *gitdir = getenv(GIT_DIR_ENVIRONMENT);
char *cwd;
int len;
enum format_type wanted = format;
enum path_format wanted = arg_path_format;
if (arg[2] == 'g') { /* --git-dir */
if (gitdir) {
print_path(gitdir, prefix, format, DEFAULT_UNMODIFIED);
print_path(gitdir, prefix, arg_path_format, PATH_FORMAT_UNMODIFIED);
continue;
}
if (!prefix) {
print_path(".git", prefix, format, DEFAULT_UNMODIFIED);
print_path(".git", prefix, arg_path_format, PATH_FORMAT_UNMODIFIED);
continue;
}
} else { /* --absolute-git-dir */
wanted = FORMAT_CANONICAL;
wanted = PATH_FORMAT_CANONICAL;
if (!gitdir && !prefix)
gitdir = ".git";
if (gitdir) {
@@ -1066,11 +1009,11 @@ int cmd_rev_parse(int argc,
strbuf_reset(&buf);
strbuf_addf(&buf, "%s%s.git", cwd, len && cwd[len-1] != '/' ? "/" : "");
free(cwd);
print_path(buf.buf, prefix, wanted, DEFAULT_CANONICAL);
print_path(buf.buf, prefix, wanted, PATH_FORMAT_CANONICAL);
continue;
}
if (!strcmp(arg, "--git-common-dir")) {
print_path(repo_get_common_dir(the_repository), prefix, format, DEFAULT_RELATIVE_IF_SHARED);
print_path(repo_get_common_dir(the_repository), prefix, arg_path_format, PATH_FORMAT_RELATIVE_IF_SHARED);
continue;
}
if (!strcmp(arg, "--is-inside-git-dir")) {
@@ -1100,7 +1043,7 @@ int cmd_rev_parse(int argc,
if (the_repository->index->split_index) {
const struct object_id *oid = &the_repository->index->split_index->base_oid;
const char *path = repo_git_path_replace(the_repository, &buf, "sharedindex.%s", oid_to_hex(oid));
print_path(path, prefix, format, DEFAULT_RELATIVE);
print_path(path, prefix, arg_path_format, PATH_FORMAT_RELATIVE);
}
continue;
}

70
path.c
View File

@@ -1579,6 +1579,76 @@ char *xdg_cache_home(const char *filename)
return NULL;
}
void append_formatted_path(struct strbuf *dest, const char *path,
const char *prefix, enum path_format format)
{
switch (format) {
case PATH_FORMAT_DEFAULT:
case PATH_FORMAT_UNMODIFIED:
strbuf_addstr(dest, path);
break;
case PATH_FORMAT_RELATIVE: {
struct strbuf relative_buf = STRBUF_INIT;
struct strbuf real_path = STRBUF_INIT;
struct strbuf real_prefix = STRBUF_INIT;
char *cwd = NULL;
/*
* We don't ever produce a relative path if prefix is NULL,
* so set the prefix to the current directory so that we can
* produce a relative path whenever possible.
*/
if (!prefix)
prefix = cwd = xgetcwd();
if (!is_absolute_path(path)) {
strbuf_realpath_forgiving(&real_path, path, 1);
path = real_path.buf;
}
if (!is_absolute_path(prefix)) {
strbuf_realpath_forgiving(&real_prefix, prefix, 1);
prefix = real_prefix.buf;
}
strbuf_addstr(dest, relative_path(path, prefix, &relative_buf));
strbuf_release(&relative_buf);
strbuf_release(&real_path);
strbuf_release(&real_prefix);
free(cwd);
break;
}
case PATH_FORMAT_RELATIVE_IF_SHARED: {
struct strbuf relative_buf = STRBUF_INIT;
/*
* If we're using RELATIVE_IF_SHARED mode, then we want an
* absolute path unless the two share a common prefix, so don't
* default the prefix to the current working directory. Doing so
* would cause a relative path to always be produced if possible.
*/
strbuf_addstr(dest, relative_path(path, prefix, &relative_buf));
strbuf_release(&relative_buf);
break;
}
case PATH_FORMAT_CANONICAL: {
struct strbuf canonical_buf = STRBUF_INIT;
strbuf_realpath_forgiving(&canonical_buf, path, 1);
strbuf_addbuf(dest, &canonical_buf);
strbuf_release(&canonical_buf);
break;
}
default:
BUG("unknown path_format value %d", format);
}
}
REPO_GIT_PATH_FUNC(squash_msg, "SQUASH_MSG")
REPO_GIT_PATH_FUNC(merge_msg, "MERGE_MSG")
REPO_GIT_PATH_FUNC(merge_rr, "MERGE_RR")

36
path.h
View File

@@ -262,6 +262,42 @@ enum scld_error safe_create_leading_directories_no_share(char *path);
int safe_create_file_with_leading_directories(struct repository *repo,
const char *path);
/**
* The formatting strategy to apply when writing a path into a buffer.
*/
enum path_format {
/*
* Represents the default formatting behavior. Treated as
* PATH_FORMAT_UNMODIFIED by append_formatted_path().
*/
PATH_FORMAT_DEFAULT,
/* Output the path exactly as-is without any modifications. */
PATH_FORMAT_UNMODIFIED,
/* Output a path relative to the provided directory prefix. */
PATH_FORMAT_RELATIVE,
/* Output a relative path only if the path shares a root with the prefix. */
PATH_FORMAT_RELATIVE_IF_SHARED,
/* Output a fully resolved, absolute canonical path. */
PATH_FORMAT_CANONICAL
};
/**
* Format a path according to the specified formatting strategy and append
* the result to the given strbuf.
*
* `dest` : The string buffer to append the formatted path to.
* `path` : The path string that needs to be formatted.
* `prefix` : The directory prefix to calculate relative offsets against.
* Pass NULL to default to the current working directory where applicable.
* `format` : The formatting behavior rule to execute.
*/
void append_formatted_path(struct strbuf *dest, const char *path,
const char *prefix, enum path_format format);
# ifdef USE_THE_REPOSITORY_VARIABLE
# include "strbuf.h"
# include "repository.h"

View File

@@ -155,4 +155,72 @@ test_expect_success 'git repo info -h shows only repo info usage' '
test_grep ! "git repo structure" actual
'
# Helper function to test path keys in both absolute and relative formats.
# $1: label for the test
# $2: field_name (e.g., commondir)
# $3: unique repo name for isolation
# $4: expect_absolute (suffix appended to repo root)
# $5: expect_relative (the relative path string expected)
# $6: init_command (extra setup like exporting env vars)
test_repo_info_path () {
label=$1
field_name=$2
repo_name=$3
expect_absolute_suffix=$4
expect_relative=$5
init_command=$6
absolute_root="$repo_name-absolute"
relative_root="$repo_name-relative"
test_expect_success "setup: $label" '
git init "$absolute_root" &&
git init "$relative_root" &&
mkdir -p "$absolute_root/sub" "$relative_root/sub"
'
test_expect_success "absolute: $label" '
(
cd "$absolute_root/sub" &&
ROOT="$(test-tool path-utils real_path ..)" && export ROOT &&
eval "$init_command" &&
expect_path="$ROOT${expect_absolute_suffix:+/$expect_absolute_suffix}" &&
echo "path.$field_name.absolute=$expect_path" >expect &&
git repo info "path.$field_name.absolute" >actual &&
test_cmp expect actual
)
'
test_expect_success "relative: $label" '
(
cd "$relative_root/sub" &&
ROOT="$(test-tool path-utils real_path ..)" && export ROOT &&
eval "$init_command" &&
echo "path.$field_name.relative=$expect_relative" >expect &&
git repo info "path.$field_name.relative" >actual &&
test_cmp expect actual
)
'
}
test_repo_info_path 'commondir standard' 'commondir' 'commondir-std' \
'.git' '../.git'
test_repo_info_path 'commondir with GIT_COMMON_DIR and GIT_DIR' 'commondir' \
'commondir-envs' 'custom-common' '../custom-common' \
'GIT_COMMON_DIR="$ROOT/custom-common" && export GIT_COMMON_DIR &&
GIT_DIR="../.git" && export GIT_DIR &&
git init --bare "$ROOT/custom-common"'
test_repo_info_path 'commondir with only GIT_DIR' 'commondir' \
'commondir-only-gitdir' '.git' '../.git' \
'GIT_DIR="../.git" && export GIT_DIR'
test_repo_info_path 'gitdir standard' 'gitdir' 'gitdir-std' \
'.git' '../.git'
test_repo_info_path 'gitdir with explicit GIT_DIR' 'gitdir' \
'gitdir-env' '.git' '../.git' \
'GIT_DIR="../.git" && export GIT_DIR'
test_done