Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(825)

Unified Diff: tests/fake_repos.py

Issue 6627013: Correctly kill 'git daemon' child process, fixing a lot of testing issues. (Closed) Base URL: svn://svn.chromium.org/chrome/trunk/tools/depot_tools
Patch Set: Created 9 years, 10 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
« no previous file with comments | « no previous file | no next file » | no next file with comments »
Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
Index: tests/fake_repos.py
diff --git a/tests/fake_repos.py b/tests/fake_repos.py
index 8674aca56acf1f5a6ea2cbe23bec013a1623ccd8..2fde1e9a51ac07fb61a3d9ff0594f69a86783c75 100755
--- a/tests/fake_repos.py
+++ b/tests/fake_repos.py
@@ -11,9 +11,11 @@ import logging
import os
import pprint
import re
+import socket
import stat
import subprocess
import sys
+import tempfile
import time
import unittest
@@ -24,23 +26,35 @@ import scm
## Utility functions
+def kill_pid(pid):
+ """Kills a process by its process id."""
+ try:
+ import signal
+ return os.kill(pid, signal.SIGKILL)
+ except ImportError:
+ pass
+
+
def addKill():
Evan Martin 2011/03/04 18:13:13 The capitalization style of this doesn't match the
M-A Ruel 2011/03/04 18:53:30 I want to convert to PEP8 but I do that incrementa
- """Add kill() method to subprocess.Popen for python <2.6"""
- if getattr(subprocess.Popen, 'kill', None):
+ """Adds kill() method to subprocess.Popen for python <2.6"""
+ if hasattr(subprocess.Popen, 'kill'):
return
+
# Unable to import 'module'
# pylint: disable=F0401
if sys.platform == 'win32':
def kill_win(process):
- import win32process
- # Access to a protected member _handle of a client class
- # pylint: disable=W0212
- return win32process.TerminateProcess(process._handle, -1)
+ try:
+ import win32process
+ # Access to a protected member _handle of a client class
+ # pylint: disable=W0212
+ return win32process.TerminateProcess(process._handle, -1)
+ except ImportError:
+ pass
Evan Martin 2011/03/04 18:13:13 Maybe split this out into a function like kill_pid
M-A Ruel 2011/03/04 18:53:30 done
subprocess.Popen.kill = kill_win
else:
def kill_nix(process):
- import signal
- return os.kill(process.pid, signal.SIGKILL)
+ return kill_pid(process.pid)
subprocess.Popen.kill = kill_nix
@@ -250,11 +264,14 @@ class FakeReposBase(object):
self.gitdaemon = None
self.common_init = False
self.repos_dir = None
+ self.gitpidfile = None
Evan Martin 2011/03/04 18:13:13 git_pid_file
self.git_root = None
self.svn_checkout = None
self.svn_repo = None
self.git_dirty = False
self.svn_dirty = False
+ self.svn_port = 3690
+ self.git_port = 9418
def trial_dir(self):
if not self.TRIAL_DIR:
@@ -281,39 +298,52 @@ class FakeReposBase(object):
def cleanup_dirt(self):
"""For each dirty repository, regenerate it."""
- if self.svnserve and self.svn_dirty:
+ if self.svn_dirty:
+ if not self.tearDownSVN():
+ logging.warning('Using both leaking checkout and svn dirty checkout')
+ if self.git_dirty:
+ if not self.tearDownGIT():
+ logging.warning('Using both leaking checkout and git dirty checkout')
+
+ def tearDown(self):
+ self.tearDownSVN()
+ self.tearDownGIT()
+ if not self.SHOULD_LEAK:
+ logging.debug('Removing %s' % self.trial_dir())
+ rmtree(self.trial_dir())
+
+ def tearDownSVN(self):
Evan Martin 2011/03/04 18:13:13 Also weird capitalization here.
+ if self.svnserve:
logging.debug('Killing svnserve pid %s' % self.svnserve.pid)
self.svnserve.kill()
+ self.wait_for_port_to_close(self.svn_port)
self.svnserve = None
if not self.SHOULD_LEAK:
- logging.debug('Removing dirty %s' % self.svn_repo)
+ logging.debug('Removing %s' % self.svn_repo)
rmtree(self.svn_repo)
- logging.debug('Removing dirty %s' % self.svn_checkout)
+ logging.debug('Removing %s' % self.svn_checkout)
rmtree(self.svn_checkout)
else:
- logging.warning('Using both leaking checkout and dirty checkout')
- if self.gitdaemon and self.git_dirty:
+ return False
+ return True
+
+ def tearDownGIT(self):
Evan Martin 2011/03/04 18:13:13 "Git" or "git", but never "GIT"
+ if self.gitdaemon:
logging.debug('Killing git-daemon pid %s' % self.gitdaemon.pid)
self.gitdaemon.kill()
self.gitdaemon = None
+ if self.gitpidfile:
+ pid = int(self.gitpidfile.read())
+ self.gitpidfile.close()
+ kill_pid(pid)
+ self.gitpidfile = None
+ self.wait_for_port_to_close(self.git_port)
if not self.SHOULD_LEAK:
- logging.debug('Removing dirty %s' % self.git_root)
+ logging.debug('Removing %s' % self.git_root)
rmtree(self.git_root)
else:
- logging.warning('Using both leaking checkout and dirty checkout')
-
- def tearDown(self):
- if self.svnserve:
- logging.debug('Killing svnserve pid %s' % self.svnserve.pid)
- self.svnserve.kill()
- self.svnserve = None
- if self.gitdaemon:
- logging.debug('Killing git-daemon pid %s' % self.gitdaemon.pid)
- self.gitdaemon.kill()
- self.gitdaemon = None
- if not self.SHOULD_LEAK:
- logging.debug('Removing %s' % self.trial_dir())
- rmtree(self.trial_dir())
+ return False
+ return True
@staticmethod
def _genTree(root, tree_dict):
@@ -354,7 +384,9 @@ class FakeReposBase(object):
cmd = ['svnserve', '-d', '--foreground', '-r', self.repos_dir]
if self.HOST == '127.0.0.1':
cmd.append('--listen-host=127.0.0.1')
+ self.check_port_is_free(self.svn_port)
self.svnserve = Popen(cmd, cwd=self.svn_repo)
+ self.wait_for_port_to_open(self.svn_port)
self.populateSvn()
self.svn_dirty = False
return True
@@ -366,16 +398,23 @@ class FakeReposBase(object):
return True
if sys.platform == 'win32':
return False
+ assert self.gitpidfile == None
for repo in ['repo_%d' % r for r in range(1, self.NB_GIT_REPOS + 1)]:
check_call(['git', 'init', '-q', join(self.git_root, repo)])
self.git_hashes[repo] = [None]
+ # Unlike svn, populate git before starting the server.
self.populateGit()
# Start the daemon.
- cmd = ['git', 'daemon', '--export-all', '--base-path=' + self.repos_dir]
+ self.gitpidfile = tempfile.NamedTemporaryFile()
+ cmd = ['git', 'daemon',
+ '--export-all',
+ '--base-path=' + self.repos_dir,
+ '--pid-file=' + self.gitpidfile.name]
if self.HOST == '127.0.0.1':
cmd.append('--listen=127.0.0.1')
- logging.debug(cmd)
+ self.check_port_is_free(self.git_port)
self.gitdaemon = Popen(cmd, cwd=self.repos_dir)
+ self.wait_for_port_to_open(self.git_port)
self.git_dirty = False
return True
@@ -400,6 +439,44 @@ class FakeReposBase(object):
new_tree = tree.copy()
self.git_hashes[repo].append((commit_hash, new_tree))
+ def check_port_is_free(self, port):
+ sock = socket.socket()
+ try:
+ sock.connect((self.HOST, port))
+ # It worked, throw.
+ logging.fatal('%d shouldn\'t be open' % port)
Evan Martin 2011/03/04 18:13:13 "open" is a confusing word here -- it means not-fr
M-A Ruel 2011/03/04 18:53:30 Renamed.
+ assert False
+ except EnvironmentError:
+ pass
+ finally:
+ sock.close()
+
+ def wait_for_port_to_open(self, port):
+ sock = socket.socket()
+ try:
+ while True:
+ try:
+ sock.connect((self.HOST, port))
+ logging.debug('%d is now open' % port)
+ break
+ except EnvironmentError:
+ pass
+ logging.debug('%d not open yet' % port)
+ finally:
+ sock.close()
Evan Martin 2011/03/04 18:13:13 Should this have a timeout int it?
M-A Ruel 2011/03/04 18:53:30 Added a clamp at 30 seconds.
+
+ def wait_for_port_to_close(self, port):
+ while True:
+ try:
+ sock = socket.socket()
+ sock.connect((self.HOST, port))
+ logging.debug('%d was open, waiting to close' % port)
Evan Martin 2011/03/04 18:13:13 Is the wait done by the sock.close()?
M-A Ruel 2011/03/04 18:53:30 There's no wait done, it loops as fast as it can,
+ except EnvironmentError:
+ logging.debug('%d now closed' % port)
+ break
+ finally:
+ sock.close()
+
def populateSvn(self):
raise NotImplementedError()
« no previous file with comments | « no previous file | no next file » | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698