Commit 50fd2bd7 authored by maruel@chromium.org's avatar maruel@chromium.org

Fix multiple tests on Windows.

BUG=23328
Review URL: http://codereview.chromium.org/2446001

git-svn-id: svn://svn.chromium.org/chrome/trunk/tools/depot_tools@48613 0039d316-1c4b-4281-b951-d872f2087c98
parent 1e3b8f0d
...@@ -6,13 +6,15 @@ ...@@ -6,13 +6,15 @@
"""Generate fake repositories for testing.""" """Generate fake repositories for testing."""
import atexit import atexit
import errno
import logging import logging
import os import os
import pprint import pprint
import re import re
import shutil import stat
import subprocess import subprocess
import sys import sys
import time
import unittest import unittest
...@@ -35,10 +37,69 @@ def addKill(): ...@@ -35,10 +37,69 @@ def addKill():
subprocess.kill = kill_nix subprocess.kill = kill_nix
def rmtree(path): def rmtree(*path):
"""Delete a directory.""" """Recursively removes a directory, even if it's marked read-only.
if os.path.exists(path):
shutil.rmtree(path) Remove the directory located at *path, if it exists.
shutil.rmtree() doesn't work on Windows if any of the files or directories
are read-only, which svn repositories and some .svn files are. We need to
be able to force the files to be writable (i.e., deletable) as we traverse
the tree.
Even with all this, Windows still sometimes fails to delete a file, citing
a permission error (maybe something to do with antivirus scans or disk
indexing). The best suggestion any of the user forums had was to wait a
bit and try again, so we do that too. It's hand-waving, but sometimes it
works. :/
"""
file_path = os.path.join(*path)
if not os.path.exists(file_path):
return
def RemoveWithRetry_win(rmfunc, path):
os.chmod(path, stat.S_IWRITE)
if win32_api_avail:
win32api.SetFileAttributes(path, win32con.FILE_ATTRIBUTE_NORMAL)
try:
return rmfunc(path)
except EnvironmentError, e:
if e.errno != errno.EACCES:
raise
print 'Failed to delete %s: trying again' % repr(path)
time.sleep(0.1)
return rmfunc(path)
def RemoveWithRetry_non_win(rmfunc, path):
if os.path.islink(path):
return os.remove(path)
else:
return rmfunc(path)
win32_api_avail = False
remove_with_retry = None
if sys.platform.startswith('win'):
# Some people don't have the APIs installed. In that case we'll do without.
try:
win32api = __import__('win32api')
win32con = __import__('win32con')
win32_api_avail = True
except ImportError:
pass
remove_with_retry = RemoveWithRetry_win
else:
remove_with_retry = RemoveWithRetry_non_win
for root, dirs, files in os.walk(file_path, topdown=False):
# For POSIX: making the directory writable guarantees removability.
# Windows will ignore the non-read-only bits in the chmod value.
os.chmod(root, 0770)
for name in files:
remove_with_retry(os.remove, os.path.join(root, name))
for name in dirs:
remove_with_retry(os.rmdir, os.path.join(root, name))
remove_with_retry(os.rmdir, file_path)
def write(path, content): def write(path, content):
...@@ -69,7 +130,9 @@ def read_tree(tree_root): ...@@ -69,7 +130,9 @@ def read_tree(tree_root):
for d in filter(lambda x: x.startswith('.'), dirs): for d in filter(lambda x: x.startswith('.'), dirs):
dirs.remove(d) dirs.remove(d)
for f in [join(root, f) for f in files if not f.startswith('.')]: for f in [join(root, f) for f in files if not f.startswith('.')]:
tree[f[len(tree_root) + 1:]] = open(join(root, f), 'rb').read() filepath = f[len(tree_root) + 1:].replace(os.sep, '/')
assert len(filepath), f
tree[filepath] = open(join(root, f), 'rU').read()
return tree return tree
...@@ -92,7 +155,7 @@ def mangle_svn_tree(*args): ...@@ -92,7 +155,7 @@ def mangle_svn_tree(*args):
for k, v in tree.iteritems(): for k, v in tree.iteritems():
if not k.startswith(old_root): if not k.startswith(old_root):
continue continue
result[join(new_root, k[len(old_root) + 1:])] = v result[join(new_root, k[len(old_root) + 1:]).replace(os.sep, '/')] = v
return result return result
...@@ -330,8 +393,10 @@ hooks = [ ...@@ -330,8 +393,10 @@ hooks = [
def setUpGIT(self): def setUpGIT(self):
"""Creates git repositories and start the servers.""" """Creates git repositories and start the servers."""
if self.gitdaemon: if self.gitdaemon:
return return True
self.setUp() self.setUp()
if sys.platform == 'win32':
return False
for repo in ['repo_%d' % r for r in range(1, 5)]: for repo in ['repo_%d' % r for r in range(1, 5)]:
check_call(['git', 'init', '-q', join(self.git_root, repo)]) check_call(['git', 'init', '-q', join(self.git_root, repo)])
self.git_hashes[repo] = [] self.git_hashes[repo] = []
...@@ -424,6 +489,7 @@ hooks = [ ...@@ -424,6 +489,7 @@ hooks = [
cmd.append('--listen=127.0.0.1') cmd.append('--listen=127.0.0.1')
logging.debug(cmd) logging.debug(cmd)
self.gitdaemon = Popen(cmd, cwd=self.repos_dir) self.gitdaemon = Popen(cmd, cwd=self.repos_dir)
return True
def _commit_svn(self, tree): def _commit_svn(self, tree):
self._genTree(self.svn_root, tree) self._genTree(self.svn_root, tree)
......
This diff is collapsed.
...@@ -750,7 +750,8 @@ class InputApiUnittest(PresubmitTestsBase): ...@@ -750,7 +750,8 @@ class InputApiUnittest(PresubmitTestsBase):
self.assertEquals(results[i].LocalPath(), self.assertEquals(results[i].LocalPath(),
presubmit.normpath(item[1][i])) presubmit.normpath(item[1][i]))
# Same number of expected results. # Same number of expected results.
self.assertEquals(sorted([f.LocalPath() for f in results]), self.assertEquals(sorted([f.LocalPath().replace(presubmit.os.sep, '/')
for f in results]),
sorted(item[1])) sorted(item[1]))
def testCustomFilter(self): def testCustomFilter(self):
......
...@@ -40,7 +40,7 @@ class TryChangeUnittest(TryChangeTestsBase): ...@@ -40,7 +40,7 @@ class TryChangeUnittest(TryChangeTestsBase):
"""General trychange.py tests.""" """General trychange.py tests."""
def testMembersChanged(self): def testMembersChanged(self):
members = [ members = [
'EscapeDot', 'GIT', 'GuessVCS', 'GetMungedDiff', 'EPILOG', 'EscapeDot', 'GIT', 'GuessVCS', 'GetMungedDiff',
'HELP_STRING', 'InvalidScript', 'NoTryServerAccess', 'PrintSuccess', 'HELP_STRING', 'InvalidScript', 'NoTryServerAccess', 'PrintSuccess',
'SCM', 'SVN', 'TryChange', 'USAGE', 'SCM', 'SVN', 'TryChange', 'USAGE',
'breakpad', 'datetime', 'errno', 'gclient_utils', 'getpass', 'json', 'breakpad', 'datetime', 'errno', 'gclient_utils', 'getpass', 'json',
......
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