mirror of
https://github.com/git/git.git
synced 2024-11-05 08:47:56 +01:00
Merge branch 'bs/lock'
* bs/lock: Add test for symlinked configuration file updates. use lockfile.c routines in git_commit_set_multivar() fully resolve symlinks when creating lockfiles
This commit is contained in:
commit
922b0e35b9
3 changed files with 134 additions and 25 deletions
30
config.c
30
config.c
|
@ -715,7 +715,7 @@ int git_config_set_multivar(const char* key, const char* value,
|
||||||
int fd = -1, in_fd;
|
int fd = -1, in_fd;
|
||||||
int ret;
|
int ret;
|
||||||
char* config_filename;
|
char* config_filename;
|
||||||
char* lock_file;
|
struct lock_file *lock = NULL;
|
||||||
const char* last_dot = strrchr(key, '.');
|
const char* last_dot = strrchr(key, '.');
|
||||||
|
|
||||||
config_filename = getenv(CONFIG_ENVIRONMENT);
|
config_filename = getenv(CONFIG_ENVIRONMENT);
|
||||||
|
@ -725,7 +725,6 @@ int git_config_set_multivar(const char* key, const char* value,
|
||||||
config_filename = git_path("config");
|
config_filename = git_path("config");
|
||||||
}
|
}
|
||||||
config_filename = xstrdup(config_filename);
|
config_filename = xstrdup(config_filename);
|
||||||
lock_file = xstrdup(mkpath("%s.lock", config_filename));
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Since "key" actually contains the section name and the real
|
* Since "key" actually contains the section name and the real
|
||||||
|
@ -770,11 +769,12 @@ int git_config_set_multivar(const char* key, const char* value,
|
||||||
store.key[i] = 0;
|
store.key[i] = 0;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* The lock_file serves a purpose in addition to locking: the new
|
* The lock serves a purpose in addition to locking: the new
|
||||||
* contents of .git/config will be written into it.
|
* contents of .git/config will be written into it.
|
||||||
*/
|
*/
|
||||||
fd = open(lock_file, O_WRONLY | O_CREAT | O_EXCL, 0666);
|
lock = xcalloc(sizeof(struct lock_file), 1);
|
||||||
if (fd < 0 || adjust_shared_perm(lock_file)) {
|
fd = hold_lock_file_for_update(lock, config_filename, 0);
|
||||||
|
if (fd < 0) {
|
||||||
fprintf(stderr, "could not lock config file\n");
|
fprintf(stderr, "could not lock config file\n");
|
||||||
free(store.key);
|
free(store.key);
|
||||||
ret = -1;
|
ret = -1;
|
||||||
|
@ -914,25 +914,31 @@ int git_config_set_multivar(const char* key, const char* value,
|
||||||
goto write_err_out;
|
goto write_err_out;
|
||||||
|
|
||||||
munmap(contents, contents_sz);
|
munmap(contents, contents_sz);
|
||||||
unlink(config_filename);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (rename(lock_file, config_filename) < 0) {
|
if (close(fd) || commit_lock_file(lock) < 0) {
|
||||||
fprintf(stderr, "Could not rename the lock file?\n");
|
fprintf(stderr, "Cannot commit config file!\n");
|
||||||
ret = 4;
|
ret = 4;
|
||||||
goto out_free;
|
goto out_free;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* fd is closed, so don't try to close it below. */
|
||||||
|
fd = -1;
|
||||||
|
/*
|
||||||
|
* lock is committed, so don't try to roll it back below.
|
||||||
|
* NOTE: Since lockfile.c keeps a linked list of all created
|
||||||
|
* lock_file structures, it isn't safe to free(lock). It's
|
||||||
|
* better to just leave it hanging around.
|
||||||
|
*/
|
||||||
|
lock = NULL;
|
||||||
ret = 0;
|
ret = 0;
|
||||||
|
|
||||||
out_free:
|
out_free:
|
||||||
if (0 <= fd)
|
if (0 <= fd)
|
||||||
close(fd);
|
close(fd);
|
||||||
|
if (lock)
|
||||||
|
rollback_lock_file(lock);
|
||||||
free(config_filename);
|
free(config_filename);
|
||||||
if (lock_file) {
|
|
||||||
unlink(lock_file);
|
|
||||||
free(lock_file);
|
|
||||||
}
|
|
||||||
return ret;
|
return ret;
|
||||||
|
|
||||||
write_err_out:
|
write_err_out:
|
||||||
|
|
114
lockfile.c
114
lockfile.c
|
@ -25,23 +25,111 @@ static void remove_lock_file_on_signal(int signo)
|
||||||
raise(signo);
|
raise(signo);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* p = absolute or relative path name
|
||||||
|
*
|
||||||
|
* Return a pointer into p showing the beginning of the last path name
|
||||||
|
* element. If p is empty or the root directory ("/"), just return p.
|
||||||
|
*/
|
||||||
|
static char *last_path_elm(char *p)
|
||||||
|
{
|
||||||
|
/* r starts pointing to null at the end of the string */
|
||||||
|
char *r = strchr(p, '\0');
|
||||||
|
|
||||||
|
if (r == p)
|
||||||
|
return p; /* just return empty string */
|
||||||
|
|
||||||
|
r--; /* back up to last non-null character */
|
||||||
|
|
||||||
|
/* back up past trailing slashes, if any */
|
||||||
|
while (r > p && *r == '/')
|
||||||
|
r--;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* then go backwards until I hit a slash, or the beginning of
|
||||||
|
* the string
|
||||||
|
*/
|
||||||
|
while (r > p && *(r-1) != '/')
|
||||||
|
r--;
|
||||||
|
return r;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/* We allow "recursive" symbolic links. Only within reason, though */
|
||||||
|
#define MAXDEPTH 5
|
||||||
|
|
||||||
|
/*
|
||||||
|
* p = path that may be a symlink
|
||||||
|
* s = full size of p
|
||||||
|
*
|
||||||
|
* If p is a symlink, attempt to overwrite p with a path to the real
|
||||||
|
* file or directory (which may or may not exist), following a chain of
|
||||||
|
* symlinks if necessary. Otherwise, leave p unmodified.
|
||||||
|
*
|
||||||
|
* This is a best-effort routine. If an error occurs, p will either be
|
||||||
|
* left unmodified or will name a different symlink in a symlink chain
|
||||||
|
* that started with p's initial contents.
|
||||||
|
*
|
||||||
|
* Always returns p.
|
||||||
|
*/
|
||||||
|
|
||||||
|
static char *resolve_symlink(char *p, size_t s)
|
||||||
|
{
|
||||||
|
int depth = MAXDEPTH;
|
||||||
|
|
||||||
|
while (depth--) {
|
||||||
|
char link[PATH_MAX];
|
||||||
|
int link_len = readlink(p, link, sizeof(link));
|
||||||
|
if (link_len < 0) {
|
||||||
|
/* not a symlink anymore */
|
||||||
|
return p;
|
||||||
|
}
|
||||||
|
else if (link_len < sizeof(link))
|
||||||
|
/* readlink() never null-terminates */
|
||||||
|
link[link_len] = '\0';
|
||||||
|
else {
|
||||||
|
warning("%s: symlink too long", p);
|
||||||
|
return p;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (link[0] == '/') {
|
||||||
|
/* absolute path simply replaces p */
|
||||||
|
if (link_len < s)
|
||||||
|
strcpy(p, link);
|
||||||
|
else {
|
||||||
|
warning("%s: symlink too long", p);
|
||||||
|
return p;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
/*
|
||||||
|
* link is a relative path, so I must replace the
|
||||||
|
* last element of p with it.
|
||||||
|
*/
|
||||||
|
char *r = (char*)last_path_elm(p);
|
||||||
|
if (r - p + link_len < s)
|
||||||
|
strcpy(r, link);
|
||||||
|
else {
|
||||||
|
warning("%s: symlink too long", p);
|
||||||
|
return p;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return p;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
static int lock_file(struct lock_file *lk, const char *path)
|
static int lock_file(struct lock_file *lk, const char *path)
|
||||||
{
|
{
|
||||||
int fd;
|
int fd;
|
||||||
struct stat st;
|
|
||||||
|
|
||||||
if ((!lstat(path, &st)) && S_ISLNK(st.st_mode)) {
|
if (strlen(path) >= sizeof(lk->filename)) return -1;
|
||||||
ssize_t sz;
|
strcpy(lk->filename, path);
|
||||||
static char target[PATH_MAX];
|
/*
|
||||||
sz = readlink(path, target, sizeof(target));
|
* subtract 5 from size to make sure there's room for adding
|
||||||
if (sz < 0)
|
* ".lock" for the lock file name
|
||||||
warning("Cannot readlink %s", path);
|
*/
|
||||||
else if (target[0] != '/')
|
resolve_symlink(lk->filename, sizeof(lk->filename)-5);
|
||||||
warning("Cannot lock target of relative symlink %s", path);
|
strcat(lk->filename, ".lock");
|
||||||
else
|
|
||||||
path = target;
|
|
||||||
}
|
|
||||||
sprintf(lk->filename, "%s.lock", path);
|
|
||||||
fd = open(lk->filename, O_RDWR | O_CREAT | O_EXCL, 0666);
|
fd = open(lk->filename, O_RDWR | O_CREAT | O_EXCL, 0666);
|
||||||
if (0 <= fd) {
|
if (0 <= fd) {
|
||||||
if (!lock_file_list) {
|
if (!lock_file_list) {
|
||||||
|
|
|
@ -595,4 +595,19 @@ echo >>result
|
||||||
|
|
||||||
test_expect_success '--null --get-regexp' 'cmp result expect'
|
test_expect_success '--null --get-regexp' 'cmp result expect'
|
||||||
|
|
||||||
|
test_expect_success 'symlinked configuration' '
|
||||||
|
|
||||||
|
ln -s notyet myconfig &&
|
||||||
|
GIT_CONFIG=myconfig git config test.frotz nitfol &&
|
||||||
|
test -h myconfig &&
|
||||||
|
test -f notyet &&
|
||||||
|
test "z$(GIT_CONFIG=notyet git config test.frotz)" = znitfol &&
|
||||||
|
GIT_CONFIG=myconfig git config test.xyzzy rezrov &&
|
||||||
|
test -h myconfig &&
|
||||||
|
test -f notyet &&
|
||||||
|
test "z$(GIT_CONFIG=notyet git config test.frotz)" = znitfol &&
|
||||||
|
test "z$(GIT_CONFIG=notyet git config test.xyzzy)" = zrezrov
|
||||||
|
|
||||||
|
'
|
||||||
|
|
||||||
test_done
|
test_done
|
||||||
|
|
Loading…
Reference in a new issue