mirror of
https://github.com/git-for-windows/git.git
synced 2026-06-16 04:07:36 -05:00
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:
@@ -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:
|
||||
+
|
||||
|
||||
@@ -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 },
|
||||
};
|
||||
|
||||
|
||||
@@ -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
70
path.c
@@ -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
36
path.h
@@ -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"
|
||||
|
||||
@@ -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
|
||||
|
||||
Reference in New Issue
Block a user