Unverified Commit 3d22394a by Patrick Steinhardt Committed by GitHub

Merge pull request #4967 from tiennou/fix/4671

Incomplete commondir support
parents c5448501 501c51b2
...@@ -438,7 +438,8 @@ typedef enum { ...@@ -438,7 +438,8 @@ typedef enum {
GIT_REPOSITORY_ITEM_HOOKS, GIT_REPOSITORY_ITEM_HOOKS,
GIT_REPOSITORY_ITEM_LOGS, GIT_REPOSITORY_ITEM_LOGS,
GIT_REPOSITORY_ITEM_MODULES, GIT_REPOSITORY_ITEM_MODULES,
GIT_REPOSITORY_ITEM_WORKTREES GIT_REPOSITORY_ITEM_WORKTREES,
GIT_REPOSITORY_ITEM__LAST
} git_repository_item_t; } git_repository_item_t;
/** /**
......
...@@ -25,6 +25,10 @@ GIT_BEGIN_DECL ...@@ -25,6 +25,10 @@ GIT_BEGIN_DECL
* Note that this is only useful if you wish to associate the repository * Note that this is only useful if you wish to associate the repository
* with a non-filesystem-backed object database and config store. * with a non-filesystem-backed object database and config store.
* *
* Caveats: since this repository has no physical location, some systems
* can fail to function properly: locations under $GIT_DIR, $GIT_COMMON_DIR,
* or $GIT_INFO_DIR are impacted.
*
* @param out The blank repository * @param out The blank repository
* @return 0 on success, or an error code * @return 0 on success, or an error code
*/ */
...@@ -39,7 +43,7 @@ GIT_EXTERN(int) git_repository_new(git_repository **out); ...@@ -39,7 +43,7 @@ GIT_EXTERN(int) git_repository_new(git_repository **out);
* There's no need to call this function directly unless you're * There's no need to call this function directly unless you're
* trying to aggressively cleanup the repo before its * trying to aggressively cleanup the repo before its
* deallocation. `git_repository_free` already performs this operation * deallocation. `git_repository_free` already performs this operation
* before deallocation the repo. * before deallocating the repo.
*/ */
GIT_EXTERN(void) git_repository__cleanup(git_repository *repo); GIT_EXTERN(void) git_repository__cleanup(git_repository *repo);
......
...@@ -306,10 +306,10 @@ static int system_attr_file( ...@@ -306,10 +306,10 @@ static int system_attr_file(
static int attr_setup(git_repository *repo, git_attr_session *attr_session) static int attr_setup(git_repository *repo, git_attr_session *attr_session)
{ {
int error = 0;
const char *workdir = git_repository_workdir(repo);
git_index *idx = NULL;
git_buf path = GIT_BUF_INIT; git_buf path = GIT_BUF_INIT;
git_index *idx = NULL;
const char *workdir;
int error = 0;
if (attr_session && attr_session->init_setup) if (attr_session && attr_session->init_setup)
return 0; return 0;
...@@ -317,42 +317,38 @@ static int attr_setup(git_repository *repo, git_attr_session *attr_session) ...@@ -317,42 +317,38 @@ static int attr_setup(git_repository *repo, git_attr_session *attr_session)
if ((error = git_attr_cache__init(repo)) < 0) if ((error = git_attr_cache__init(repo)) < 0)
return error; return error;
/* preload attribute files that could contain macros so the /*
* definitions will be available for later file parsing * Preload attribute files that could contain macros so the
* definitions will be available for later file parsing.
*/ */
error = system_attr_file(&path, attr_session); if ((error = system_attr_file(&path, attr_session)) < 0 ||
(error = preload_attr_file(repo, attr_session, GIT_ATTR_FILE__FROM_FILE,
if (error == 0) NULL, path.ptr)) < 0) {
error = preload_attr_file( if (error != GIT_ENOTFOUND)
repo, attr_session, GIT_ATTR_FILE__FROM_FILE, NULL, path.ptr); goto out;
}
if (error != GIT_ENOTFOUND)
goto out;
if ((error = preload_attr_file(
repo, attr_session, GIT_ATTR_FILE__FROM_FILE,
NULL, git_repository_attr_cache(repo)->cfg_attr_file)) < 0)
goto out;
if ((error = git_repository_item_path(&path, if ((error = preload_attr_file(repo, attr_session, GIT_ATTR_FILE__FROM_FILE,
repo, GIT_REPOSITORY_ITEM_INFO)) < 0) NULL, git_repository_attr_cache(repo)->cfg_attr_file)) < 0)
goto out; goto out;
if ((error = preload_attr_file( if ((error = git_repository_item_path(&path, repo, GIT_REPOSITORY_ITEM_INFO)) < 0 ||
repo, attr_session, GIT_ATTR_FILE__FROM_FILE, (error = preload_attr_file(repo, attr_session, GIT_ATTR_FILE__FROM_FILE,
path.ptr, GIT_ATTR_FILE_INREPO)) < 0) path.ptr, GIT_ATTR_FILE_INREPO)) < 0) {
goto out; if (error != GIT_ENOTFOUND)
goto out;
}
if (workdir != NULL && if ((workdir = git_repository_workdir(repo)) != NULL &&
(error = preload_attr_file( (error = preload_attr_file(repo, attr_session, GIT_ATTR_FILE__FROM_FILE,
repo, attr_session, GIT_ATTR_FILE__FROM_FILE, workdir, GIT_ATTR_FILE)) < 0) workdir, GIT_ATTR_FILE)) < 0)
goto out; goto out;
if ((error = git_repository_index__weakptr(&idx, repo)) < 0 || if ((error = git_repository_index__weakptr(&idx, repo)) < 0 ||
(error = preload_attr_file( (error = preload_attr_file(repo, attr_session, GIT_ATTR_FILE__FROM_INDEX,
repo, attr_session, GIT_ATTR_FILE__FROM_INDEX, NULL, GIT_ATTR_FILE)) < 0) NULL, GIT_ATTR_FILE)) < 0)
goto out; goto out;
if (attr_session) if (attr_session)
attr_session->init_setup = 1; attr_session->init_setup = 1;
...@@ -516,15 +512,12 @@ static int collect_attr_files( ...@@ -516,15 +512,12 @@ static int collect_attr_files(
* - $GIT_PREFIX/etc/gitattributes * - $GIT_PREFIX/etc/gitattributes
*/ */
error = git_repository_item_path(&attrfile, repo, GIT_REPOSITORY_ITEM_INFO); if ((error = git_repository_item_path(&attrfile, repo, GIT_REPOSITORY_ITEM_INFO)) < 0 ||
if (error < 0) (error = push_attr_file(repo, attr_session, files, GIT_ATTR_FILE__FROM_FILE,
goto cleanup; attrfile.ptr, GIT_ATTR_FILE_INREPO)) < 0) {
if (error != GIT_ENOTFOUND)
error = push_attr_file( goto cleanup;
repo, attr_session, files, GIT_ATTR_FILE__FROM_FILE, }
attrfile.ptr, GIT_ATTR_FILE_INREPO);
if (error < 0)
goto cleanup;
info.repo = repo; info.repo = repo;
info.attr_session = attr_session; info.attr_session = attr_session;
......
...@@ -335,16 +335,13 @@ int git_ignore__for_path( ...@@ -335,16 +335,13 @@ int git_ignore__for_path(
goto cleanup; goto cleanup;
} }
if ((error = git_repository_item_path(&infopath, /* load .git/info/exclude if possible */
repo, GIT_REPOSITORY_ITEM_INFO)) < 0) if ((error = git_repository_item_path(&infopath, repo, GIT_REPOSITORY_ITEM_INFO)) < 0 ||
goto cleanup; (error = push_ignore_file(ignores, &ignores->ign_global, infopath.ptr, GIT_IGNORE_FILE_INREPO)) < 0) {
if (error != GIT_ENOTFOUND)
/* load .git/info/exclude */ goto cleanup;
error = push_ignore_file( error = 0;
ignores, &ignores->ign_global, }
infopath.ptr, GIT_IGNORE_FILE_INREPO);
if (error < 0)
goto cleanup;
/* load core.excludesfile */ /* load core.excludesfile */
if (git_repository_attr_cache(repo)->cfg_excl_file != NULL) if (git_repository_attr_cache(repo)->cfg_excl_file != NULL)
......
...@@ -548,6 +548,7 @@ int git_odb__add_default_backends( ...@@ -548,6 +548,7 @@ int git_odb__add_default_backends(
#else #else
if (p_stat(objects_dir, &st) < 0) { if (p_stat(objects_dir, &st) < 0) {
if (as_alternates) if (as_alternates)
/* this should warn */
return 0; return 0;
git_error_set(GIT_ERROR_ODB, "failed to load object database in '%s'", objects_dir); git_error_set(GIT_ERROR_ODB, "failed to load object database in '%s'", objects_dir);
......
...@@ -71,6 +71,7 @@ int git_odb__hashobj(git_oid *id, git_rawobj *obj); ...@@ -71,6 +71,7 @@ int git_odb__hashobj(git_oid *id, git_rawobj *obj);
* Format the object header such as it would appear in the on-disk object * Format the object header such as it would appear in the on-disk object
*/ */
int git_odb__format_object_header(size_t *out_len, char *hdr, size_t hdr_size, git_off_t obj_len, git_object_t obj_type); int git_odb__format_object_header(size_t *out_len, char *hdr, size_t hdr_size, git_off_t obj_len, git_object_t obj_type);
/* /*
* Hash an open file descriptor. * Hash an open file descriptor.
* This is a performance call when the contents of a fd need to be hashed, * This is a performance call when the contents of a fd need to be hashed,
...@@ -95,7 +96,7 @@ int git_odb__hashfd_filtered( ...@@ -95,7 +96,7 @@ int git_odb__hashfd_filtered(
* symlink, then the raw contents of the symlink will be hashed. Otherwise, * symlink, then the raw contents of the symlink will be hashed. Otherwise,
* this will fallback to `git_odb__hashfd`. * this will fallback to `git_odb__hashfd`.
* *
* The hash type for this call is always `GIT_OBJIECT_BLOB` because * The hash type for this call is always `GIT_OBJECT_BLOB` because
* symlinks may only point to blobs. * symlinks may only point to blobs.
*/ */
int git_odb__hashlink(git_oid *out, const char *path); int git_odb__hashlink(git_oid *out, const char *path);
......
...@@ -42,23 +42,24 @@ bool git_repository__fsync_gitdir = false; ...@@ -42,23 +42,24 @@ bool git_repository__fsync_gitdir = false;
static const struct { static const struct {
git_repository_item_t parent; git_repository_item_t parent;
git_repository_item_t fallback;
const char *name; const char *name;
bool directory; bool directory;
} items[] = { } items[] = {
{ GIT_REPOSITORY_ITEM_GITDIR, NULL, true }, { GIT_REPOSITORY_ITEM_GITDIR, GIT_REPOSITORY_ITEM__LAST, NULL, true },
{ GIT_REPOSITORY_ITEM_WORKDIR, NULL, true }, { GIT_REPOSITORY_ITEM_WORKDIR, GIT_REPOSITORY_ITEM__LAST, NULL, true },
{ GIT_REPOSITORY_ITEM_COMMONDIR, NULL, true }, { GIT_REPOSITORY_ITEM_COMMONDIR, GIT_REPOSITORY_ITEM__LAST, NULL, true },
{ GIT_REPOSITORY_ITEM_GITDIR, "index", false }, { GIT_REPOSITORY_ITEM_GITDIR, GIT_REPOSITORY_ITEM__LAST, "index", false },
{ GIT_REPOSITORY_ITEM_COMMONDIR, "objects", true }, { GIT_REPOSITORY_ITEM_COMMONDIR, GIT_REPOSITORY_ITEM_GITDIR, "objects", true },
{ GIT_REPOSITORY_ITEM_COMMONDIR, "refs", true }, { GIT_REPOSITORY_ITEM_COMMONDIR, GIT_REPOSITORY_ITEM_GITDIR, "refs", true },
{ GIT_REPOSITORY_ITEM_COMMONDIR, "packed-refs", false }, { GIT_REPOSITORY_ITEM_COMMONDIR, GIT_REPOSITORY_ITEM_GITDIR, "packed-refs", false },
{ GIT_REPOSITORY_ITEM_COMMONDIR, "remotes", true }, { GIT_REPOSITORY_ITEM_COMMONDIR, GIT_REPOSITORY_ITEM_GITDIR, "remotes", true },
{ GIT_REPOSITORY_ITEM_COMMONDIR, "config", false }, { GIT_REPOSITORY_ITEM_COMMONDIR, GIT_REPOSITORY_ITEM_GITDIR, "config", false },
{ GIT_REPOSITORY_ITEM_COMMONDIR, "info", true }, { GIT_REPOSITORY_ITEM_COMMONDIR, GIT_REPOSITORY_ITEM_GITDIR, "info", true },
{ GIT_REPOSITORY_ITEM_COMMONDIR, "hooks", true }, { GIT_REPOSITORY_ITEM_COMMONDIR, GIT_REPOSITORY_ITEM_GITDIR, "hooks", true },
{ GIT_REPOSITORY_ITEM_COMMONDIR, "logs", true }, { GIT_REPOSITORY_ITEM_COMMONDIR, GIT_REPOSITORY_ITEM_GITDIR, "logs", true },
{ GIT_REPOSITORY_ITEM_GITDIR, "modules", true }, { GIT_REPOSITORY_ITEM_GITDIR, GIT_REPOSITORY_ITEM__LAST, "modules", true },
{ GIT_REPOSITORY_ITEM_COMMONDIR, "worktrees", true } { GIT_REPOSITORY_ITEM_COMMONDIR, GIT_REPOSITORY_ITEM_GITDIR, "worktrees", true }
}; };
static int check_repositoryformatversion(git_config *config); static int check_repositoryformatversion(git_config *config);
...@@ -2308,11 +2309,11 @@ int git_repository_is_empty(git_repository *repo) ...@@ -2308,11 +2309,11 @@ int git_repository_is_empty(git_repository *repo)
return is_empty; return is_empty;
} }
int git_repository_item_path(git_buf *out, const git_repository *repo, git_repository_item_t item) static const char *resolved_parent_path(const git_repository *repo, git_repository_item_t item, git_repository_item_t fallback)
{ {
const char *parent; const char *parent;
switch (items[item].parent) { switch (item) {
case GIT_REPOSITORY_ITEM_GITDIR: case GIT_REPOSITORY_ITEM_GITDIR:
parent = git_repository_path(repo); parent = git_repository_path(repo);
break; break;
...@@ -2324,9 +2325,17 @@ int git_repository_item_path(git_buf *out, const git_repository *repo, git_repos ...@@ -2324,9 +2325,17 @@ int git_repository_item_path(git_buf *out, const git_repository *repo, git_repos
break; break;
default: default:
git_error_set(GIT_ERROR_INVALID, "invalid item directory"); git_error_set(GIT_ERROR_INVALID, "invalid item directory");
return -1; return NULL;
} }
if (!parent && fallback != GIT_REPOSITORY_ITEM__LAST)
return resolved_parent_path(repo, fallback, GIT_REPOSITORY_ITEM__LAST);
return parent;
}
int git_repository_item_path(git_buf *out, const git_repository *repo, git_repository_item_t item)
{
const char *parent = resolved_parent_path(repo, items[item].parent, items[item].fallback);
if (parent == NULL) { if (parent == NULL) {
git_error_set(GIT_ERROR_INVALID, "path cannot exist in repository"); git_error_set(GIT_ERROR_INVALID, "path cannot exist in repository");
return GIT_ENOTFOUND; return GIT_ENOTFOUND;
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment