On Fri, Apr 12, 2013 at 12:16:00PM -0400, Jeff King wrote:
> One option we have not explored is an environment variable
> to loosen git's requirement. I'm thinking something like
> GIT_INACCESSIBLE_HOMEDIR_OK, which could be set by default
> when git-daemon uses --user.
>
> That would leave all existing setups working, but would
> still enable the extra protections for people not running
> git-daemon (and people who use git via sudo could choose to
> set it, too, if they would prefer that to setting up HOME).
So here's what I came up with. I tried to make the exception as tight as
possible by checking that $HOME was actually the problem, as that is the
common problem (you switch users, but HOME is pointing to the old user).
It means that we would still die if ~root is readable, but
~root/.gitconfig is not (or ~root/.config is not). I think that is
probably a good thing, though, as it is an indication of something more
interesting going on that the user should investigate.
Given how tight the exception is, I almost wonder if we should drop the
environment variable completely, and just never complain about this case
(in other words, just make it a loosening of 96b9e0e3).
-Peff
---
diff --git a/Documentation/git.txt b/Documentation/git.txt
index 6a875f2..e004ee8 100644
--- a/Documentation/git.txt
+++ b/Documentation/git.txt
@@ -806,6 +806,15 @@ for further details.
temporarily to avoid using a buggy `/etc/gitconfig` file while
waiting for someone with sufficient permissions to fix it.
+'GIT_INACCESSIBLE_HOME_OK'::
+ Normally git will complain and die if it cannot access
+ `$HOME/.gitconfig`. If this variable is set to "1", then
+ a `$HOME` directory which cannot be accessed will not be
+ considered an error. This can be useful if you are switching
+ user ids without resetting `$HOME`, and are willing to take the
+ risk that some configuration options might not be used (because
+ git could not read the config file that contained them).
+
'GIT_FLUSH'::
If this environment variable is set to "1", then commands such
as 'git blame' (in incremental mode), 'git rev-list', 'git log',
diff --git a/cache.h b/cache.h
index e1e8ce8..01f300f 100644
--- a/cache.h
+++ b/cache.h
@@ -358,6 +358,7 @@ static inline enum object_type object_type(unsigned int
mode)
#define GIT_NOTES_REWRITE_REF_ENVIRONMENT "GIT_NOTES_REWRITE_REF"
#define GIT_NOTES_REWRITE_MODE_ENVIRONMENT "GIT_NOTES_REWRITE_MODE"
#define GIT_LITERAL_PATHSPECS_ENVIRONMENT "GIT_LITERAL_PATHSPECS"
+#define GIT_INACCESSIBLE_HOME_OK_ENVIRONMENT "GIT_INACCESSIBLE_HOME_OK"
/*
* This environment variable is expected to contain a boolean indicating
diff --git a/daemon.c b/daemon.c
index 131b049..6c56cc0 100644
--- a/daemon.c
+++ b/daemon.c
@@ -1091,7 +1091,7 @@ static void drop_privileges(struct credentials *cred)
if (cred && (initgroups(cred->pass->pw_name, cred->gid) ||
setgid (cred->gid) || setuid(cred->pass->pw_uid)))
die("cannot drop privileges");
- setenv("GIT_CONFIG_INACCESSIBLE_HOME_OK", "1", 0);
+ setenv(GIT_INACCESSIBLE_HOME_OK_ENVIRONMENT, "1", 0);
}
static struct credentials *prepare_credentials(const char *user_name,
diff --git a/wrapper.c b/wrapper.c
index bac59d2..644f867 100644
--- a/wrapper.c
+++ b/wrapper.c
@@ -416,11 +416,52 @@ int access_or_die(const char *path, int mode)
return ret;
}
+/*
+ * Returns true iff the path is under $HOME, that directory is inaccessible,
+ * and the user has told us through the environment that an inaccessible HOME
+ * is OK. The results are cached so that repeated calls will not make multiple
+ * syscalls.
+ */
+static int inaccessible_home_ok(const char *path)
+{
+ static int gentle = -1;
+ static int home_inaccessible = -1;
+ const char *home;
+
+ if (gentle < 0)
+ gentle = git_env_bool(GIT_INACCESSIBLE_HOME_OK_ENVIRONMENT, 0);
+ if (!gentle)
+ return 0;
+
+ home = getenv("HOME");
+ if (!home)
+ return 0;
+ /*
+ * We do not bother with normalizing PATHs to avoid symlinks
+ * here, since the point is to catch paths that are
+ * constructed as "$HOME/...".
+ */
+ if (prefixcmp(path, home) && path[strlen(home)] == '/')
+ return 0;
+
+ if (home_inaccessible < 0)
+ home_inaccessible = access(home, R_OK|X_OK) < 0 && errno ==
EACCES;
+ return home_inaccessible;
+}
+
int access_or_die(const char *path, int mode)
{
int ret = access(path, mode);
- if (ret && errno != ENOENT && errno != ENOTDIR)
+ if (ret && errno != ENOENT && errno != ENOTDIR) {
+ /*
+ * We do not have to worry about clobbering errno
+ * in inaccessible_home_ok, because we get either EACCES
+ * again, or we die.
+ */
+ if (errno == EACCES && inaccessible_home_ok(path))
+ return ret;
die_errno(_("unable to access '%s'"), path);
+ }
return ret;
}
>
> -Peff
--
To unsubscribe from this list: send the line "unsubscribe git" in
the body of a message to [email protected]
More majordomo info at http://vger.kernel.org/majordomo-info.html