Commit 07ab60e2 authored by maruel@chromium.org's avatar maruel@chromium.org

Move reverting code from gclient_scm to scm to be able to reuse the code.

There is 2 changes while moving the code:
- Making externals handling an option.
- Not trapping EnvironmentError.

This code will be reused by the commit queue.

TEST=unit tests
BUG=none

Review URL: http://codereview.chromium.org/6250177

git-svn-id: svn://svn.chromium.org/chrome/trunk/tools/depot_tools@74181 0039d316-1c4b-4281-b951-d872f2087c98
parent 082cccdb
...@@ -838,49 +838,22 @@ class SVNWrapper(SCMWrapper): ...@@ -838,49 +838,22 @@ class SVNWrapper(SCMWrapper):
# Don't reuse the args. # Don't reuse the args.
return self.update(options, [], file_list) return self.update(options, [], file_list)
for file_status in scm.SVN.CaptureStatus(self.checkout_path): def printcb(file_status):
file_path = os.path.join(self.checkout_path, file_status[1]) file_list.append(file_status[1])
# Temporarily forcibly delete externals to make sure chromium can build
# without svn:external's.
#if file_status[0][0] == 'X':
# # Ignore externals.
# logging.info('Ignoring external %s' % file_path)
# continue
if logging.getLogger().isEnabledFor(logging.INFO): if logging.getLogger().isEnabledFor(logging.INFO):
logging.info('%s%s' % (file[0], file[1])) logging.info('%s%s' % (file_status[0], file_status[1]))
else: else:
print(file_path) print(os.path.join(self.checkout_path, file_status[1]))
scm.SVN.Revert(self.checkout_path, callback=printcb)
if file_status[0].isspace():
logging.error('No idea what is the status of %s.\n'
'You just found a bug in gclient, please ping '
'maruel@chromium.org ASAP!' % file_path)
# svn revert is really stupid. It fails on inconsistent line-endings,
# on switched directories, etc. So take no chance and delete everything!
try:
if not os.path.exists(file_path):
pass
elif os.path.isfile(file_path) or os.path.islink(file_path):
logging.info('os.remove(%s)' % file_path)
os.remove(file_path)
elif os.path.isdir(file_path):
logging.info('gclient_utils.RemoveDirectory(%s)' % file_path)
gclient_utils.RemoveDirectory(file_path)
else:
logging.error('no idea what is %s.\nYou just found a bug in gclient'
', please ping maruel@chromium.org ASAP!' % file_path)
except EnvironmentError:
logging.error('Failed to remove %s.' % file_path)
try: try:
# svn revert is so broken we don't even use it. Using # svn revert is so broken we don't even use it. Using
# "svn up --revision BASE" achieve the same effect. # "svn up --revision BASE" achieve the same effect.
# file_list will contain duplicates.
self._RunAndGetFileList(['update', '--revision', 'BASE'], options, self._RunAndGetFileList(['update', '--revision', 'BASE'], options,
file_list) file_list)
except OSError, e: except OSError, e:
# Maybe the directory disapeared meanwhile. We don't want it to throw an # Maybe the directory disapeared meanwhile. Do not throw an exception.
# exception.
logging.error('Failed to update:\n%s' % str(e)) logging.error('Failed to update:\n%s' % str(e))
def revinfo(self, options, args, file_list): def revinfo(self, options, args, file_list):
......
...@@ -237,7 +237,7 @@ def CheckChangeHasNoTabs(input_api, output_api, source_file_filter=None): ...@@ -237,7 +237,7 @@ def CheckChangeHasNoTabs(input_api, output_api, source_file_filter=None):
def CheckChangeTodoHasOwner(input_api, output_api, source_file_filter=None): def CheckChangeTodoHasOwner(input_api, output_api, source_file_filter=None):
"""Checks that the user didn't add TODO(name) without an owner.""" """Checks that the user didn't add TODO(name) without an owner."""
unowned_todo = input_api.re.compile('TO' + 'DO[^(]'); unowned_todo = input_api.re.compile('TO' + 'DO[^(]')
for f, line_num, line in input_api.RightHandSideLines(source_file_filter): for f, line_num, line in input_api.RightHandSideLines(source_file_filter):
if unowned_todo.search(line): if unowned_todo.search(line):
text = ('Found TO' + 'DO with no owner in %s, line %s' % text = ('Found TO' + 'DO with no owner in %s, line %s' %
......
...@@ -6,6 +6,7 @@ ...@@ -6,6 +6,7 @@
import cStringIO import cStringIO
import glob import glob
import logging
import os import os
import re import re
import shutil import shutil
...@@ -828,3 +829,48 @@ class SVN(object): ...@@ -828,3 +829,48 @@ class SVN(object):
elif ver > min_ver: elif ver > min_ver:
return (True, SVN.current_version) return (True, SVN.current_version)
return (True, SVN.current_version) return (True, SVN.current_version)
@staticmethod
def Revert(repo_root, callback=None, ignore_externals=False):
"""Reverts all svn modifications in repo_root, including properties.
Deletes any modified files or directory.
A "svn update --revision BASE" call is required after to revive deleted
files.
"""
for file_status in SVN.CaptureStatus(repo_root):
file_path = os.path.join(repo_root, file_status[1])
if ignore_externals and file_status[0][0] == 'X':
# Ignore externals.
logging.info('Ignoring external %s' % file_status[1])
continue
if callback:
callback(file_status)
if file_status[0].isspace():
# Try reverting the file since it's probably a property change.
gclient_utils.CheckCall(
['svn', 'revert', file_status[1]], cwd=repo_root)
# svn revert is really stupid. It fails on inconsistent line-endings,
# on switched directories, etc. So take no chance and delete everything!
if file_status[0][0] == 'D':
# Deleted file requires manual intervention and require calling
# revert, like for properties.
gclient_utils.CheckCall(
['svn', 'revert', file_status[1]], cwd=repo_root)
else:
if not os.path.exists(file_path):
pass
elif os.path.isfile(file_path) or os.path.islink(file_path):
logging.info('os.remove(%s)' % file_path)
os.remove(file_path)
elif os.path.isdir(file_path):
logging.info('gclient_utils.RemoveDirectory(%s)' % file_path)
gclient_utils.RemoveDirectory(file_path)
else:
logging.critical(
('No idea what is %s.\nYou just found a bug in gclient'
', please ping maruel@chromium.org ASAP!') % file_path)
...@@ -38,7 +38,7 @@ class RootTestCase(BaseSCMTestCase): ...@@ -38,7 +38,7 @@ class RootTestCase(BaseSCMTestCase):
self.mox.ReplayAll() self.mox.ReplayAll()
members = [ members = [
'GetCasedPath', 'GenFakeDiff', 'GIT', 'SVN', 'ValidateEmail', 'GetCasedPath', 'GenFakeDiff', 'GIT', 'SVN', 'ValidateEmail',
'cStringIO', 'gclient_utils', 'glob', 'os', 're', 'shutil', 'cStringIO', 'gclient_utils', 'glob', 'logging', 'os', 're', 'shutil',
'subprocess', 'sys', 'tempfile', 'time', 'xml', 'subprocess', 'sys', 'tempfile', 'time', 'xml',
] ]
# If this test fails, you should add the relevant test. # If this test fails, you should add the relevant test.
...@@ -77,7 +77,7 @@ class SVNTestCase(BaseSCMTestCase): ...@@ -77,7 +77,7 @@ class SVNTestCase(BaseSCMTestCase):
'AssertVersion', 'Capture', 'CaptureRevision', 'CaptureInfo', 'AssertVersion', 'Capture', 'CaptureRevision', 'CaptureInfo',
'CaptureStatus', 'current_version', 'DiffItem', 'GenerateDiff', 'CaptureStatus', 'current_version', 'DiffItem', 'GenerateDiff',
'GetCheckoutRoot', 'GetEmail', 'GetFileProperty', 'IsMoved', 'GetCheckoutRoot', 'GetEmail', 'GetFileProperty', 'IsMoved',
'IsMovedInfo', 'ReadSimpleAuth', 'RunAndGetFileList', 'IsMovedInfo', 'ReadSimpleAuth', 'Revert', 'RunAndGetFileList',
] ]
# If this test fails, you should add the relevant test. # If this test fails, you should add the relevant test.
self.compareMembers(scm.SVN, members) self.compareMembers(scm.SVN, members)
......
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