Commit 0db557ce authored by pkasting@chromium.org's avatar pkasting@chromium.org

Reland r195308, r195328, and r195363, which use --internal-diff on svn 1.7+.

This time, do not use the --config-dir hack on svn < 1.7 for the diff command
that runs against a remote URL.  This seems to trigger auth prompts (see revert
comments on https://codereview.chromium.org/14297017/ ).

Review URL: https://chromiumcodereview.appspot.com/14084009

git-svn-id: svn://svn.chromium.org/chrome/trunk/tools/depot_tools@196263 0039d316-1c4b-4281-b951-d872f2087c98
parent ee10d7d7
...@@ -781,87 +781,106 @@ class SVN(object): ...@@ -781,87 +781,106 @@ class SVN(object):
The diff will always use relative paths. The diff will always use relative paths.
""" """
assert isinstance(filenames, (list, tuple)) assert isinstance(filenames, (list, tuple))
# If the user specified a custom diff command in their svn config file,
# then it'll be used when we do svn diff, which we don't want to happen
# since we want the unified diff.
if SVN.AssertVersion("1.7")[0]:
# On svn >= 1.7, the "--internal-diff" flag will solve this.
return SVN._GenerateDiffInternal(filenames, cwd, full_move, revision,
["diff", "--internal-diff"],
["diff", "--internal-diff"])
else:
# On svn < 1.7, the "--internal-diff" flag doesn't exist. Using
# --diff-cmd=diff doesn't always work, since e.g. Windows cmd users may
# not have a "diff" executable in their path at all. So we use an empty
# temporary directory as the config directory, which bypasses any user
# settings for the diff-cmd. However, we don't pass this for the
# remote_safe_diff_command parameter, since when a new config-dir is
# specified for an svn diff against a remote URL, it triggers
# authentication prompts. In this case there isn't really a good
# alternative to svn 1.7's --internal-diff flag.
bogus_dir = tempfile.mkdtemp()
try:
return SVN._GenerateDiffInternal(filenames, cwd, full_move, revision,
["diff", "--config-dir", bogus_dir],
["diff"])
finally:
gclient_utils.rmtree(bogus_dir)
@staticmethod
def _GenerateDiffInternal(filenames, cwd, full_move, revision, diff_command,
remote_safe_diff_command):
root = os.path.normcase(os.path.join(cwd, '')) root = os.path.normcase(os.path.join(cwd, ''))
def RelativePath(path, root): def RelativePath(path, root):
"""We must use relative paths.""" """We must use relative paths."""
if os.path.normcase(path).startswith(root): if os.path.normcase(path).startswith(root):
return path[len(root):] return path[len(root):]
return path return path
# If the user specified a custom diff command in their svn config file, # Cleanup filenames
# then it'll be used when we do svn diff, which we don't want to happen filenames = [RelativePath(f, root) for f in filenames]
# since we want the unified diff. Using --diff-cmd=diff doesn't always # Get information about the modified items (files and directories)
# work, since they can have another diff executable in their path that data = dict((f, SVN.CaptureLocalInfo([f], root)) for f in filenames)
# gives different line endings. So we use a bogus temp directory as the diffs = []
# config directory, which gets around these problems. if full_move:
bogus_dir = tempfile.mkdtemp() # Eliminate modified files inside moved/copied directory.
try: for (filename, info) in data.iteritems():
# Cleanup filenames if SVN.IsMovedInfo(info) and info.get("Node Kind") == "directory":
filenames = [RelativePath(f, root) for f in filenames] # Remove files inside the directory.
# Get information about the modified items (files and directories) filenames = [f for f in filenames
data = dict([(f, SVN.CaptureLocalInfo([f], root)) for f in filenames]) if not f.startswith(filename + os.path.sep)]
diffs = [] for filename in data.keys():
if full_move: if not filename in filenames:
# Eliminate modified files inside moved/copied directory. # Remove filtered out items.
for (filename, info) in data.iteritems(): del data[filename]
if SVN.IsMovedInfo(info) and info.get("Node Kind") == "directory": else:
# Remove files inside the directory. metaheaders = []
filenames = [f for f in filenames for (filename, info) in data.iteritems():
if not f.startswith(filename + os.path.sep)] if SVN.IsMovedInfo(info):
for filename in data.keys(): # for now, the most common case is a head copy,
if not filename in filenames: # so let's just encode that as a straight up cp.
# Remove filtered out items. srcurl = info.get('Copied From URL')
del data[filename] file_root = info.get('Repository Root')
else: rev = int(info.get('Copied From Rev'))
metaheaders = [] assert srcurl.startswith(file_root)
for (filename, info) in data.iteritems(): src = srcurl[len(file_root)+1:]
if SVN.IsMovedInfo(info): try:
# for now, the most common case is a head copy, srcinfo = SVN.CaptureRemoteInfo(srcurl)
# so let's just encode that as a straight up cp. except subprocess2.CalledProcessError, e:
srcurl = info.get('Copied From URL') if not 'Not a valid URL' in e.stderr:
file_root = info.get('Repository Root') raise
rev = int(info.get('Copied From Rev')) # Assume the file was deleted. No idea how to figure out at which
assert srcurl.startswith(file_root) # revision the file was deleted.
src = srcurl[len(file_root)+1:] srcinfo = {'Revision': rev}
try: if (srcinfo.get('Revision') != rev and
srcinfo = SVN.CaptureRemoteInfo(srcurl) SVN.Capture(remote_safe_diff_command + ['-r', '%d:head' % rev,
except subprocess2.CalledProcessError, e: srcurl], cwd)):
if not 'Not a valid URL' in e.stderr: metaheaders.append("#$ svn cp -r %d %s %s "
raise "### WARNING: note non-trunk copy\n" %
# Assume the file was deleted. No idea how to figure out at which (rev, src, filename))
# revision the file was deleted. else:
srcinfo = {'Revision': rev} metaheaders.append("#$ cp %s %s\n" % (src,
if (srcinfo.get('Revision') != rev and filename))
SVN.Capture(['diff', '-r', '%d:head' % rev, srcurl], cwd)): if metaheaders:
metaheaders.append("#$ svn cp -r %d %s %s " diffs.append("### BEGIN SVN COPY METADATA\n")
"### WARNING: note non-trunk copy\n" % diffs.extend(metaheaders)
(rev, src, filename)) diffs.append("### END SVN COPY METADATA\n")
else: # Now ready to do the actual diff.
metaheaders.append("#$ cp %s %s\n" % (src, for filename in sorted(data):
filename)) diffs.append(SVN._DiffItemInternal(
filename, cwd, data[filename], diff_command, full_move, revision))
if metaheaders: # Use StringIO since it can be messy when diffing a directory move with
diffs.append("### BEGIN SVN COPY METADATA\n") # full_move=True.
diffs.extend(metaheaders) buf = cStringIO.StringIO()
diffs.append("### END SVN COPY METADATA\n") for d in filter(None, diffs):
# Now ready to do the actual diff. buf.write(d)
for filename in sorted(data.iterkeys()): result = buf.getvalue()
diffs.append(SVN._DiffItemInternal( buf.close()
filename, cwd, data[filename], bogus_dir, full_move, revision)) return result
# Use StringIO since it can be messy when diffing a directory move with
# full_move=True.
buf = cStringIO.StringIO()
for d in filter(None, diffs):
buf.write(d)
result = buf.getvalue()
buf.close()
return result
finally:
gclient_utils.rmtree(bogus_dir)
@staticmethod @staticmethod
def _DiffItemInternal(filename, cwd, info, bogus_dir, full_move, revision): def _DiffItemInternal(filename, cwd, info, diff_command, full_move, revision):
"""Grabs the diff data.""" """Grabs the diff data."""
command = ["diff", "--config-dir", bogus_dir, filename] command = diff_command + [filename]
if revision: if revision:
command.extend(['--revision', revision]) command.extend(['--revision', revision])
data = None data = None
......
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