commit - 0a706d22c0eedf209fc5cc5821872ac9b47ecfd4
commit + 611e5fc2074d428e17f920dc595496af4dd0dc77
blob - 28d77bb457c59dd0fa56e815055a611a3c6fdd07
blob + cc1772cd735799b69ba788a42ce58bc0214cbc96
--- lib/got_lib_privsep.h
+++ lib/got_lib_privsep.h
*/
struct got_imsg_patch {
int git;
+ int xbit;
char old[PATH_MAX];
char new[PATH_MAX];
char cid[41];
blob - 4b795cc39551259dc64fe67a92381bb2c0591143
blob + f8e3aaf634fc474c5546bddf69cefd58eb36f969
--- lib/patch.c
+++ lib/patch.c
STAILQ_HEAD(got_patch_hunk_head, got_patch_hunk);
struct got_patch {
+ int xbit;
char *old;
char *new;
char cid[41];
if (*patch.blob != '\0')
strlcpy(p->blob, patch.blob, sizeof(p->blob));
+
+ p->xbit = patch.xbit;
/* automatically set strip=1 for git-style diffs */
if (strip == -1 && patch.git &&
goto done;
}
mode = sb.st_mode;
- }
+ } else if (p->xbit)
+ mode |= (S_IXUSR | S_IXGRP | S_IXOTH);
err = got_opentemp_named(&tmppath, &tmpfile, template);
if (err)
blob - 73996be84412da8b6da8967a3b861c8e404900d5
blob + b01acc2d4192df91e119abcea319ac890c164de0
--- libexec/got-read-patch/got-read-patch.c
+++ libexec/got-read-patch/got-read-patch.c
static const struct got_error *
send_patch(const char *oldname, const char *newname, const char *commitid,
- const char *blob, int git)
+ const char *blob, const int xbit, int git)
{
struct got_imsg_patch p;
if (blob != NULL)
strlcpy(p.blob, blob, sizeof(p.blob));
+ p.xbit = xbit;
p.git = git;
if (imsg_compose(&ibuf, GOT_IMSG_PATCH, 0, 0, -1, &p, sizeof(p)) == -1)
return got_error_from_errno("imsg_compose GOT_IMSG_PATCH");
return NULL;
}
+static int
+filexbit(const char *line)
+{
+ char *m;
+
+ m = strchr(line, '(');
+ if (m && !strncmp(m + 1, "mode ", 5))
+ return strncmp(m + 6, "755", 3) == 0;
+
+ return 0;
+}
+
static const struct got_error *
blobid(const char *line, char **blob, int git)
{
char *line = NULL;
size_t linesize = 0;
ssize_t linelen;
- int create, rename = 0;
+ int create, rename = 0, xbit = 0;
*done = 0;
*next = 0;
} else if (!strncmp(line, "+++ ", 4)) {
free(new);
err = filename(line+4, &new);
+ } else if (!strncmp(line, "blob + ", 7)) {
+ xbit = filexbit(line);
} else if (!git && !strncmp(line, "blob - ", 7)) {
free(blob);
err = blobid(line + 7, &blob, git);
err = filename(line + 10, &new);
} else if (git && !strncmp(line, "similarity index 100%", 21))
rename = 1;
+ else if (git && !strncmp(line, "new file mode 100", 17))
+ xbit = strncmp(line + 17, "755", 3) == 0;
else if (git && !strncmp(line, "index ", 6)) {
free(blob);
err = blobid(line + 6, &blob, git);
if (rename && old != NULL && new != NULL) {
*done = 1;
err = send_patch(old, new, commitid,
- blob, git);
+ blob, xbit, git);
break;
}
err = got_error(GOT_ERR_PATCH_MALFORMED);
else
err = send_patch(old, new, commitid,
- blob, git);
+ blob, xbit, git);
if (err)
break;
blob - 1854b36c0fa1e88ead530f05e4b4011f623a055c
blob + 1bcdb584a653439ff381dc63d0dc88145ee8fa36
--- regress/cmdline/patch.sh
+++ regress/cmdline/patch.sh
ret=$?
if [ $ret -ne 0 ]; then
diff -u $testroot/wt/numbers $testroot/wt/numbers.expected
+ fi
+ test_done $testroot $ret
+}
+
+test_patch_newfile_xbit_got_diff() {
+ local testroot=`test_init patch_newfile_xbit`
+
+ got checkout $testroot/repo $testroot/wt > /dev/null
+ ret=$?
+ if [ $ret -ne 0 ]; then
+ test_done $testroot $ret
+ return 1
+ fi
+
+ cat <<EOF > $testroot/wt/patch
+blob - /dev/null
+blob + abcdef0123456789abcdef012345678901234567 (mode 755)
+--- /dev/null
++++ xfile
+@@ -0,0 +1,1 @@
++xfile
+EOF
+
+ (cd $testroot/wt && got patch patch) > /dev/null
+ ret=$?
+ if [ $ret -ne 0 ]; then
+ test_done $testroot $ret
+ return 1
fi
+
+ if [ ! -x $testroot/wt/xfile ]; then
+ echo "failed to set xbit on newfile" >&2
+ test_done $testroot 1
+ return 1
+ fi
+
+ echo xfile > $testroot/wt/xfile.expected
+ cmp -s $testroot/wt/xfile $testroot/wt/xfile.expected
+ ret=$?
+ if [ $ret -ne 0 ]; then
+ echo "fail"
+ diff -u $testroot/wt/xfile $testroot/wt/xfile.expected
+ fi
+
test_done $testroot $ret
}
+test_patch_newfile_xbit_git_diff() {
+ local testroot=`test_init patch_newfile_xbit`
+
+ got checkout $testroot/repo $testroot/wt > /dev/null
+ ret=$?
+ if [ $ret -ne 0 ]; then
+ test_done $testroot $ret
+ return 1
+ fi
+
+ cat <<EOF > $testroot/wt/patch
+diff --git a/xfile b/xfile
+new file mode 100755
+index 00000000..abcdef01
+--- /dev/null
++++ b/xfile
+@@ -0,0 +1,1 @@
++xfile
+EOF
+
+ (cd $testroot/wt && got patch patch) > /dev/null
+ ret=$?
+ if [ $ret -ne 0 ]; then
+ test_done $testroot $ret
+ return 1
+ fi
+
+ if [ ! -x $testroot/wt/xfile ]; then
+ echo "failed to set xbit on newfile" >&2
+ test_done $testroot 1
+ return 1
+ fi
+
+ echo xfile > $testroot/wt/xfile.expected
+ cmp -s $testroot/wt/xfile $testroot/wt/xfile.expected
+ ret=$?
+ if [ $ret -ne 0 ]; then
+ echo "fail"
+ diff -u $testroot/wt/xfile $testroot/wt/xfile.expected
+ fi
+
+ test_done $testroot $ret
+}
+
test_parseargs "$@"
run_test test_patch_basic
run_test test_patch_dont_apply
run_test test_patch_merge_conflict
run_test test_patch_merge_unknown_blob
run_test test_patch_merge_reverse
+run_test test_patch_newfile_xbit_got_diff
+run_test test_patch_newfile_xbit_git_diff