Merge branch 'maint'

* maint:
  http: treat config options sslCAPath and sslCAInfo as paths
  Documentation/diff: give --word-diff-regex=. example
  filter-branch: deal with object name vs. pathname ambiguity in tree-filter
  check-ignore: correct documentation about output
  git-p4: clean up after p4 submit failure
  git-p4: work with a detached head
  git-p4: add option to system() to return subshell status
  git-p4: add failing test for submit from detached head
  remote-http(s): support SOCKS proxies
  t5813: avoid creating urls that break on cygwin
  Escape Git's exec path in contrib/rerere-train.sh script
  allow hooks to ignore their standard input stream
  rebase-i-exec: Allow space in SHELL_PATH
  Documentation: make environment variable formatting more consistent
This commit is contained in:
Jeff King 2015-12-01 17:32:38 -05:00
commit 40fdcc5357
15 changed files with 137 additions and 79 deletions

View File

@ -267,6 +267,9 @@ expression to make sure that it matches all non-whitespace characters.
A match that contains a newline is silently truncated(!) at the A match that contains a newline is silently truncated(!) at the
newline. newline.
+ +
For example, `--word-diff-regex=.` will treat each character as a word
and, correspondingly, show differences character by character.
+
The regex can also be set via a diff driver or configuration option, see The regex can also be set via a diff driver or configuration option, see
linkgit:gitattributes[1] or linkgit:git-config[1]. Giving it explicitly linkgit:gitattributes[1] or linkgit:git-config[1]. Giving it explicitly
overrides any diff driver or configuration setting. Diff drivers overrides any diff driver or configuration setting. Diff drivers

View File

@ -16,10 +16,9 @@ DESCRIPTION
----------- -----------
For each pathname given via the command-line or from a file via For each pathname given via the command-line or from a file via
`--stdin`, show the pattern from .gitignore (or other input files to `--stdin`, check whether the file is excluded by .gitignore (or other
the exclude mechanism) that decides if the pathname is excluded or input files to the exclude mechanism) and output the path if it is
included. Later patterns within a file take precedence over earlier excluded.
ones.
By default, tracked files are not shown at all since they are not By default, tracked files are not shown at all since they are not
subject to exclude rules; but see `--no-index'. subject to exclude rules; but see `--no-index'.
@ -32,7 +31,8 @@ OPTIONS
-v, --verbose:: -v, --verbose::
Also output details about the matching pattern (if any) Also output details about the matching pattern (if any)
for each given pathname. for each given pathname. For precedence rules within and
between exclude sources, see linkgit:gitignore[5].
--stdin:: --stdin::
Read pathnames from the standard input, one per line, Read pathnames from the standard input, one per line,

View File

@ -1056,7 +1056,7 @@ of clones and fetches.
cloning of shallow repositories. cloning of shallow repositories.
See 'GIT_TRACE' for available trace output options. See 'GIT_TRACE' for available trace output options.
GIT_LITERAL_PATHSPECS:: 'GIT_LITERAL_PATHSPECS'::
Setting this variable to `1` will cause Git to treat all Setting this variable to `1` will cause Git to treat all
pathspecs literally, rather than as glob patterns. For example, pathspecs literally, rather than as glob patterns. For example,
running `GIT_LITERAL_PATHSPECS=1 git log -- '*.c'` will search running `GIT_LITERAL_PATHSPECS=1 git log -- '*.c'` will search
@ -1065,15 +1065,15 @@ GIT_LITERAL_PATHSPECS::
literal paths to Git (e.g., paths previously given to you by literal paths to Git (e.g., paths previously given to you by
`git ls-tree`, `--raw` diff output, etc). `git ls-tree`, `--raw` diff output, etc).
GIT_GLOB_PATHSPECS:: 'GIT_GLOB_PATHSPECS'::
Setting this variable to `1` will cause Git to treat all Setting this variable to `1` will cause Git to treat all
pathspecs as glob patterns (aka "glob" magic). pathspecs as glob patterns (aka "glob" magic).
GIT_NOGLOB_PATHSPECS:: 'GIT_NOGLOB_PATHSPECS'::
Setting this variable to `1` will cause Git to treat all Setting this variable to `1` will cause Git to treat all
pathspecs as literal (aka "literal" magic). pathspecs as literal (aka "literal" magic).
GIT_ICASE_PATHSPECS:: 'GIT_ICASE_PATHSPECS'::
Setting this variable to `1` will cause Git to treat all Setting this variable to `1` will cause Git to treat all
pathspecs as case-insensitive. pathspecs as case-insensitive.
@ -1087,7 +1087,7 @@ GIT_ICASE_PATHSPECS::
variable when it is invoked as the top level command by the variable when it is invoked as the top level command by the
end user, to be recorded in the body of the reflog. end user, to be recorded in the body of the reflog.
`GIT_REF_PARANOIA`:: 'GIT_REF_PARANOIA'::
If set to `1`, include broken or badly named refs when iterating If set to `1`, include broken or badly named refs when iterating
over lists of refs. In a normal, non-corrupted repository, this over lists of refs. In a normal, non-corrupted repository, this
does nothing. However, enabling it may help git to detect and does nothing. However, enabling it may help git to detect and
@ -1098,7 +1098,7 @@ GIT_ICASE_PATHSPECS::
an operation has touched every ref (e.g., because you are an operation has touched every ref (e.g., because you are
cloning a repository to make a backup). cloning a repository to make a backup).
`GIT_ALLOW_PROTOCOL`:: 'GIT_ALLOW_PROTOCOL'::
If set, provide a colon-separated list of protocols which are If set, provide a colon-separated list of protocols which are
allowed to be used with fetch/push/clone. This is useful to allowed to be used with fetch/push/clone. This is useful to
restrict recursive submodule initialization from an untrusted restrict recursive submodule initialization from an untrusted

View File

@ -32,6 +32,7 @@
#include "sequencer.h" #include "sequencer.h"
#include "notes-utils.h" #include "notes-utils.h"
#include "mailmap.h" #include "mailmap.h"
#include "sigchain.h"
static const char * const builtin_commit_usage[] = { static const char * const builtin_commit_usage[] = {
N_("git commit [<options>] [--] <pathspec>..."), N_("git commit [<options>] [--] <pathspec>..."),
@ -1537,8 +1538,10 @@ static int run_rewrite_hook(const unsigned char *oldsha1,
return code; return code;
n = snprintf(buf, sizeof(buf), "%s %s\n", n = snprintf(buf, sizeof(buf), "%s %s\n",
sha1_to_hex(oldsha1), sha1_to_hex(newsha1)); sha1_to_hex(oldsha1), sha1_to_hex(newsha1));
sigchain_push(SIGPIPE, SIG_IGN);
write_in_full(proc.in, buf, n); write_in_full(proc.in, buf, n);
close(proc.in); close(proc.in);
sigchain_pop(SIGPIPE);
return finish_command(&proc); return finish_command(&proc);
} }

View File

@ -7,7 +7,7 @@ USAGE="$me rev-list-args"
SUBDIRECTORY_OK=Yes SUBDIRECTORY_OK=Yes
OPTIONS_SPEC= OPTIONS_SPEC=
. $(git --exec-path)/git-sh-setup . "$(git --exec-path)/git-sh-setup"
require_work_tree require_work_tree
cd_to_toplevel cd_to_toplevel

View File

@ -349,7 +349,7 @@ while read commit parents; do
die "tree filter failed: $filter_tree" die "tree filter failed: $filter_tree"
( (
git diff-index -r --name-only --ignore-submodules $commit && git diff-index -r --name-only --ignore-submodules $commit -- &&
git ls-files --others git ls-files --others
) > "$tempdir"/tree-state || exit ) > "$tempdir"/tree-state || exit
git update-index --add --replace --remove --stdin \ git update-index --add --replace --remove --stdin \

View File

@ -203,14 +203,16 @@ def p4_has_move_command():
# assume it failed because @... was invalid changelist # assume it failed because @... was invalid changelist
return True return True
def system(cmd): def system(cmd, ignore_error=False):
expand = isinstance(cmd,basestring) expand = isinstance(cmd,basestring)
if verbose: if verbose:
sys.stderr.write("executing %s\n" % str(cmd)) sys.stderr.write("executing %s\n" % str(cmd))
retcode = subprocess.call(cmd, shell=expand) retcode = subprocess.call(cmd, shell=expand)
if retcode: if retcode and not ignore_error:
raise CalledProcessError(retcode, cmd) raise CalledProcessError(retcode, cmd)
return retcode
def p4_system(cmd): def p4_system(cmd):
"""Specifically invoke p4 as the system command. """ """Specifically invoke p4 as the system command. """
real_cmd = p4_build_cmd(cmd) real_cmd = p4_build_cmd(cmd)
@ -553,7 +555,12 @@ def p4Where(depotPath):
return clientPath return clientPath
def currentGitBranch(): def currentGitBranch():
return read_pipe("git name-rev HEAD").split(" ")[1].strip() retcode = system(["git", "symbolic-ref", "-q", "HEAD"], ignore_error=True)
if retcode != 0:
# on a detached head
return None
else:
return read_pipe(["git", "name-rev", "HEAD"]).split(" ")[1].strip()
def isValidGitDir(path): def isValidGitDir(path):
if (os.path.exists(path + "/HEAD") if (os.path.exists(path + "/HEAD")
@ -1741,44 +1748,47 @@ class P4Submit(Command, P4UserMap):
# #
# Let the user edit the change description, then submit it. # Let the user edit the change description, then submit it.
# #
if self.edit_template(fileName): submitted = False
# read the edited message and submit
ret = True
tmpFile = open(fileName, "rb")
message = tmpFile.read()
tmpFile.close()
if self.isWindows:
message = message.replace("\r\n", "\n")
submitTemplate = message[:message.index(separatorLine)]
p4_write_pipe(['submit', '-i'], submitTemplate)
if self.preserveUser: try:
if p4User: if self.edit_template(fileName):
# Get last changelist number. Cannot easily get it from # read the edited message and submit
# the submit command output as the output is tmpFile = open(fileName, "rb")
# unmarshalled. message = tmpFile.read()
changelist = self.lastP4Changelist() tmpFile.close()
self.modifyChangelistUser(changelist, p4User) if self.isWindows:
message = message.replace("\r\n", "\n")
submitTemplate = message[:message.index(separatorLine)]
p4_write_pipe(['submit', '-i'], submitTemplate)
# The rename/copy happened by applying a patch that created a if self.preserveUser:
# new file. This leaves it writable, which confuses p4. if p4User:
for f in pureRenameCopy: # Get last changelist number. Cannot easily get it from
p4_sync(f, "-f") # the submit command output as the output is
# unmarshalled.
changelist = self.lastP4Changelist()
self.modifyChangelistUser(changelist, p4User)
else: # The rename/copy happened by applying a patch that created a
# new file. This leaves it writable, which confuses p4.
for f in pureRenameCopy:
p4_sync(f, "-f")
submitted = True
finally:
# skip this patch # skip this patch
ret = False if not submitted:
print "Submission cancelled, undoing p4 changes." print "Submission cancelled, undoing p4 changes."
for f in editedFiles: for f in editedFiles:
p4_revert(f) p4_revert(f)
for f in filesToAdd: for f in filesToAdd:
p4_revert(f) p4_revert(f)
os.remove(f) os.remove(f)
for f in filesToDelete: for f in filesToDelete:
p4_revert(f) p4_revert(f)
os.remove(fileName) os.remove(fileName)
return ret return submitted
# Export git tags as p4 labels. Create a p4 label and then tag # Export git tags as p4 labels. Create a p4 label and then tag
# with that. # with that.
@ -1854,8 +1864,6 @@ class P4Submit(Command, P4UserMap):
def run(self, args): def run(self, args):
if len(args) == 0: if len(args) == 0:
self.master = currentGitBranch() self.master = currentGitBranch()
if len(self.master) == 0 or not gitBranchExists("refs/heads/%s" % self.master):
die("Detecting current git branch failed!")
elif len(args) == 1: elif len(args) == 1:
self.master = args[0] self.master = args[0]
if not branchExists(self.master): if not branchExists(self.master):
@ -1863,9 +1871,10 @@ class P4Submit(Command, P4UserMap):
else: else:
return False return False
allowSubmit = gitConfig("git-p4.allowSubmit") if self.master:
if len(allowSubmit) > 0 and not self.master in allowSubmit.split(","): allowSubmit = gitConfig("git-p4.allowSubmit")
die("%s is not in git-p4.allowSubmit" % self.master) if len(allowSubmit) > 0 and not self.master in allowSubmit.split(","):
die("%s is not in git-p4.allowSubmit" % self.master)
[upstream, settings] = findUpstreamBranchPoint() [upstream, settings] = findUpstreamBranchPoint()
self.depotPath = settings['depot-paths'][0] self.depotPath = settings['depot-paths'][0]
@ -1933,7 +1942,12 @@ class P4Submit(Command, P4UserMap):
self.check() self.check()
commits = [] commits = []
for line in read_pipe_lines(["git", "rev-list", "--no-merges", "%s..%s" % (self.origin, self.master)]): if self.master:
commitish = self.master
else:
commitish = 'HEAD'
for line in read_pipe_lines(["git", "rev-list", "--no-merges", "%s..%s" % (self.origin, commitish)]):
commits.append(line.strip()) commits.append(line.strip())
commits.reverse() commits.reverse()

View File

@ -610,7 +610,7 @@ do_next () {
read -r command rest < "$todo" read -r command rest < "$todo"
mark_action_done mark_action_done
printf 'Executing: %s\n' "$rest" printf 'Executing: %s\n' "$rest"
${SHELL:-@SHELL_PATH@} -c "$rest" # Actual execution "${SHELL:-@SHELL_PATH@}" -c "$rest" # Actual execution
status=$? status=$?
# Run in subshell because require_clean_work_tree can die. # Run in subshell because require_clean_work_tree can die.
dirty=f dirty=f

15
http.c
View File

@ -214,10 +214,10 @@ static int http_options(const char *var, const char *value, void *cb)
#endif #endif
#if LIBCURL_VERSION_NUM >= 0x070908 #if LIBCURL_VERSION_NUM >= 0x070908
if (!strcmp("http.sslcapath", var)) if (!strcmp("http.sslcapath", var))
return git_config_string(&ssl_capath, var, value); return git_config_pathname(&ssl_capath, var, value);
#endif #endif
if (!strcmp("http.sslcainfo", var)) if (!strcmp("http.sslcainfo", var))
return git_config_string(&ssl_cainfo, var, value); return git_config_pathname(&ssl_cainfo, var, value);
if (!strcmp("http.sslcertpasswordprotected", var)) { if (!strcmp("http.sslcertpasswordprotected", var)) {
ssl_cert_password_required = git_config_bool(var, value); ssl_cert_password_required = git_config_bool(var, value);
return 0; return 0;
@ -464,6 +464,17 @@ static CURL *get_curl_handle(void)
if (curl_http_proxy) { if (curl_http_proxy) {
curl_easy_setopt(result, CURLOPT_PROXY, curl_http_proxy); curl_easy_setopt(result, CURLOPT_PROXY, curl_http_proxy);
#if LIBCURL_VERSION_NUM >= 0x071800
if (starts_with(curl_http_proxy, "socks5"))
curl_easy_setopt(result,
CURLOPT_PROXYTYPE, CURLPROXY_SOCKS5);
else if (starts_with(curl_http_proxy, "socks4a"))
curl_easy_setopt(result,
CURLOPT_PROXYTYPE, CURLPROXY_SOCKS4A);
else if (starts_with(curl_http_proxy, "socks"))
curl_easy_setopt(result,
CURLOPT_PROXYTYPE, CURLPROXY_SOCKS4);
#endif
} }
#if LIBCURL_VERSION_NUM >= 0x070a07 #if LIBCURL_VERSION_NUM >= 0x070a07
curl_easy_setopt(result, CURLOPT_PROXYAUTH, CURLAUTH_ANY); curl_easy_setopt(result, CURLOPT_PROXYAUTH, CURLAUTH_ANY);

View File

@ -109,23 +109,20 @@ test_expect_success 'push to URL' '
diff expected actual diff expected actual
' '
# Test that filling pipe buffers doesn't cause failure test_expect_success 'set up many-ref tests' '
# Too slow to leave enabled for general use {
if false nr=1000
then while test $nr -lt 2000
printf 'parent1\nrepo1\n' >expected do
nr=1000 nr=$(( $nr + 1 ))
while test $nr -lt 2000 echo "create refs/heads/b/$nr $COMMIT3"
do done
nr=$(( $nr + 1 )) } | git update-ref --stdin
git branch b/$nr $COMMIT3 '
echo "refs/heads/b/$nr $COMMIT3 refs/heads/b/$nr $_z40" >>expected
done
test_expect_success 'push many refs' ' test_expect_success 'sigpipe does not cause pre-push hook failure' '
git push parent1 "refs/heads/b/*:refs/heads/b/*" && echo "exit 0" | write_script "$HOOK" &&
diff expected actual git push parent1 "refs/heads/b/*:refs/heads/b/*"
' '
fi
test_done test_done

View File

@ -14,7 +14,7 @@ test_expect_success 'setup repository to clone' '
' '
test_proto "host:path" ssh "remote:repo.git" test_proto "host:path" ssh "remote:repo.git"
test_proto "ssh://" ssh "ssh://remote/$PWD/remote/repo.git" test_proto "ssh://" ssh "ssh://remote$PWD/remote/repo.git"
test_proto "git+ssh://" ssh "git+ssh://remote/$PWD/remote/repo.git" test_proto "git+ssh://" ssh "git+ssh://remote$PWD/remote/repo.git"
test_done test_done

View File

@ -418,4 +418,11 @@ test_expect_success 'filter commit message without trailing newline' '
test_cmp expect actual test_cmp expect actual
' '
test_expect_success 'tree-filter deals with object name vs pathname ambiguity' '
test_when_finished "git reset --hard original" &&
ambiguous=$(git rev-list -1 HEAD) &&
git filter-branch --tree-filter "mv file.t $ambiguous" HEAD^.. &&
git show HEAD:$ambiguous
'
test_done test_done

View File

@ -241,6 +241,22 @@ test_expect_success 'unresolvable host in P4PORT should display error' '
) )
' '
test_expect_success 'submit from detached head' '
test_when_finished cleanup_git &&
git p4 clone --dest="$git" //depot &&
(
cd "$git" &&
git checkout p4/master &&
>detached_head_test &&
git add detached_head_test &&
git commit -m "add detached_head" &&
git config git-p4.skipSubmitEdit true &&
git p4 submit &&
git p4 rebase &&
git log p4/master | grep detached_head
)
'
test_expect_success 'kill p4d' ' test_expect_success 'kill p4d' '
kill_p4d kill_p4d
' '

View File

@ -389,7 +389,7 @@ test_expect_success 'description with Jobs section and bogus following text' '
( (
cd "$cli" && cd "$cli" &&
p4 revert desc6 && p4 revert desc6 &&
rm desc6 rm -f desc6
) )
' '

View File

@ -15,6 +15,7 @@
#include "submodule.h" #include "submodule.h"
#include "string-list.h" #include "string-list.h"
#include "sha1-array.h" #include "sha1-array.h"
#include "sigchain.h"
/* rsync support */ /* rsync support */
@ -1127,6 +1128,8 @@ static int run_pre_push_hook(struct transport *transport,
return -1; return -1;
} }
sigchain_push(SIGPIPE, SIG_IGN);
strbuf_init(&buf, 256); strbuf_init(&buf, 256);
for (r = remote_refs; r; r = r->next) { for (r = remote_refs; r; r = r->next) {
@ -1140,8 +1143,10 @@ static int run_pre_push_hook(struct transport *transport,
r->peer_ref->name, sha1_to_hex(r->new_sha1), r->peer_ref->name, sha1_to_hex(r->new_sha1),
r->name, sha1_to_hex(r->old_sha1)); r->name, sha1_to_hex(r->old_sha1));
if (write_in_full(proc.in, buf.buf, buf.len) != buf.len) { if (write_in_full(proc.in, buf.buf, buf.len) < 0) {
ret = -1; /* We do not mind if a hook does not read all refs. */
if (errno != EPIPE)
ret = -1;
break; break;
} }
} }
@ -1152,6 +1157,8 @@ static int run_pre_push_hook(struct transport *transport,
if (!ret) if (!ret)
ret = x; ret = x;
sigchain_pop(SIGPIPE);
x = finish_command(&proc); x = finish_command(&proc);
if (!ret) if (!ret)
ret = x; ret = x;