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

Side by Side Diff: tests/fake_repos.py

Issue 6625018: Add trial_dir to manage automatic trial directory creation and deletion. (Closed) Base URL: svn://svn.chromium.org/chrome/trunk/tools/depot_tools
Patch Set: Created 9 years, 9 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 unified diff | Download patch | Annotate | Revision Log
« no previous file with comments | « no previous file | tests/trial_dir.py » ('j') | tests/trial_dir.py » ('J')
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
OLDNEW
1 #!/usr/bin/python 1 #!/usr/bin/python
2 # Copyright (c) 2010 The Chromium Authors. All rights reserved. 2 # Copyright (c) 2010 The Chromium Authors. All rights reserved.
3 # Use of this source code is governed by a BSD-style license that can be 3 # Use of this source code is governed by a BSD-style license that can be
4 # found in the LICENSE file. 4 # found in the LICENSE file.
5 5
6 """Generate fake repositories for testing.""" 6 """Generate fake repositories for testing."""
7 7
8 import atexit 8 import atexit
9 import datetime 9 import datetime
10 import errno 10 import errno
11 import logging 11 import logging
12 import os 12 import os
13 import pprint 13 import pprint
14 import re 14 import re
15 import socket 15 import socket
16 import stat 16 import stat
17 import subprocess 17 import subprocess
18 import sys 18 import sys
19 import tempfile 19 import tempfile
20 import time 20 import time
21 import unittest
22 21
23 sys.path.append(os.path.dirname(os.path.dirname(os.path.abspath(__file__)))) 22 # trial_dir must be first for non-system libraries.
24 23 from tests import trial_dir
25 import scm 24 import scm
26 25
27 ## Utility functions 26 ## Utility functions
28 27
29 28
30 def kill_pid(pid): 29 def kill_pid(pid):
31 """Kills a process by its process id.""" 30 """Kills a process by its process id."""
32 try: 31 try:
33 # Unable to import 'module' 32 # Unable to import 'module'
34 # pylint: disable=F0401 33 # pylint: disable=F0401
(...skipping 178 matching lines...) Expand 10 before | Expand all | Expand 10 after
213 check_call(['git', 'add', '-A', '-f'], cwd=repo) 212 check_call(['git', 'add', '-A', '-f'], cwd=repo)
214 check_call(['git', 'commit', '-q', '--message', 'foo'], cwd=repo) 213 check_call(['git', 'commit', '-q', '--message', 'foo'], cwd=repo)
215 rev = Popen(['git', 'show-ref', '--head', 'HEAD'], 214 rev = Popen(['git', 'show-ref', '--head', 'HEAD'],
216 cwd=repo).communicate()[0].split(' ', 1)[0] 215 cwd=repo).communicate()[0].split(' ', 1)[0]
217 logging.debug('At revision %s' % rev) 216 logging.debug('At revision %s' % rev)
218 return rev 217 return rev
219 218
220 219
221 _FAKE_LOADED = False 220 _FAKE_LOADED = False
222 221
223 class FakeReposBase(object): 222 class FakeReposBase(trial_dir.TrialDir):
224 """Generate both svn and git repositories to test gclient functionality. 223 """Generate both svn and git repositories to test gclient functionality.
225 224
226 Many DEPS functionalities need to be tested: Var, File, From, deps_os, hooks, 225 Many DEPS functionalities need to be tested: Var, File, From, deps_os, hooks,
227 use_relative_paths. 226 use_relative_paths.
228 227
229 And types of dependencies: Relative urls, Full urls, both svn and git. 228 And types of dependencies: Relative urls, Full urls, both svn and git.
230 229
231 populateSvn() and populateGit() need to be implemented by the subclass. 230 populateSvn() and populateGit() need to be implemented by the subclass.
232 """ 231 """
233
234 # When SHOULD_LEAK is set to True, temporary directories created while the
235 # tests are running aren't deleted at the end of the tests. Expect failures
236 # when running more than one test due to inter-test side-effects. Helps with
237 # debugging.
238 SHOULD_LEAK = False
239 # Override if unhappy.
240 TRIAL_DIR = None
241 # Hostname 232 # Hostname
242 NB_GIT_REPOS = 1 233 NB_GIT_REPOS = 1
243 USERS = [ 234 USERS = [
244 ('user1@example.com', 'foo'), 235 ('user1@example.com', 'foo'),
245 ('user2@example.com', 'bar'), 236 ('user2@example.com', 'bar'),
246 ] 237 ]
247 238
248 def __init__(self, trial_dir=None, leak=None, host=None): 239 def __init__(self, host=None):
249 global _FAKE_LOADED 240 global _FAKE_LOADED
250 if _FAKE_LOADED: 241 if _FAKE_LOADED:
251 raise Exception('You can only start one FakeRepos at a time.') 242 raise Exception('You can only start one FakeRepos at a time.')
252 _FAKE_LOADED = True 243 _FAKE_LOADED = True
253 # Quick hack. 244
254 if '-v' in sys.argv: 245 super(FakeReposBase, self).__init__('repos')
255 logging.basicConfig(level=logging.DEBUG) 246
256 elif leak is not None:
257 self.SHOULD_LEAK = leak
258 self.host = host or '127.0.0.1' 247 self.host = host or '127.0.0.1'
259 if trial_dir:
260 self.TRIAL_DIR = trial_dir
261
262 # Format is [ None, tree, tree, ...] 248 # Format is [ None, tree, tree, ...]
263 # i.e. revisions are 1-based. 249 # i.e. revisions are 1-based.
264 self.svn_revs = [None] 250 self.svn_revs = [None]
265 # Format is { repo: [ None, (hash, tree), (hash, tree), ... ], ... } 251 # Format is { repo: [ None, (hash, tree), (hash, tree), ... ], ... }
266 # so reference looks like self.git_hashes[repo][rev][0] for hash and 252 # so reference looks like self.git_hashes[repo][rev][0] for hash and
267 # self.git_hashes[repo][rev][1] for it's tree snapshot. 253 # self.git_hashes[repo][rev][1] for it's tree snapshot.
268 # For consistency with self.svn_revs, it is 1-based too. 254 # For consistency with self.svn_revs, it is 1-based too.
269 self.git_hashes = {} 255 self.git_hashes = {}
270 self.svnserve = None 256 self.svnserve = None
271 self.gitdaemon = None 257 self.gitdaemon = None
272 self.common_init = False
273 self.repos_dir = None
274 self.git_pid_file = None 258 self.git_pid_file = None
275 self.git_root = None 259 self.git_root = None
276 self.svn_checkout = None 260 self.svn_checkout = None
277 self.svn_repo = None 261 self.svn_repo = None
278 self.git_dirty = False 262 self.git_dirty = False
279 self.svn_dirty = False 263 self.svn_dirty = False
280 self.svn_base = 'svn://%s/svn/' % self.host 264 self.svn_base = 'svn://%s/svn/' % self.host
281 self.git_base = 'git://%s/git/' % self.host 265 self.git_base = 'git://%s/git/' % self.host
282 self.svn_port = 3690 266 self.svn_port = 3690
283 self.git_port = 9418 267 self.git_port = 9418
284 268
285 def trial_dir(self):
286 if not self.TRIAL_DIR:
287 self.TRIAL_DIR = os.path.join(
288 os.path.dirname(os.path.abspath(__file__)), '_trial')
289 return self.TRIAL_DIR
290
291 def set_up(self): 269 def set_up(self):
292 """All late initialization comes here. 270 """All late initialization comes here."""
293
294 Note that it deletes all trial_dir() and not only repos_dir.
295 """
296 self.cleanup_dirt() 271 self.cleanup_dirt()
297 if not self.common_init: 272 if not self.root_dir:
298 self.common_init = True 273 try:
299 self.repos_dir = os.path.join(self.trial_dir(), 'repos') 274 add_kill()
300 self.git_root = join(self.repos_dir, 'git') 275 # self.root_dir is not set before this call.
301 self.svn_checkout = join(self.repos_dir, 'svn_checkout') 276 super(FakeReposBase, self).set_up()
Evan Martin 2011/03/04 19:46:11 I feel like this mechanism, where you conditionall
302 self.svn_repo = join(self.repos_dir, 'svn') 277 self.git_root = join(self.root_dir, 'git')
303 add_kill() 278 self.svn_checkout = join(self.root_dir, 'svn_checkout')
304 rmtree(self.trial_dir()) 279 self.svn_repo = join(self.root_dir, 'svn')
305 os.makedirs(self.repos_dir) 280 finally:
306 atexit.register(self.tear_down) 281 # Registers cleanup.
282 atexit.register(self.tear_down)
307 283
308 def cleanup_dirt(self): 284 def cleanup_dirt(self):
309 """For each dirty repository, destroy it.""" 285 """For each dirty repository, destroy it."""
310 if self.svn_dirty: 286 if self.svn_dirty:
311 if not self.tear_down_svn(): 287 if not self.tear_down_svn():
312 logging.error('Using both leaking checkout and svn dirty checkout') 288 logging.error('Using both leaking checkout and svn dirty checkout')
313 if self.git_dirty: 289 if self.git_dirty:
314 if not self.tear_down_git(): 290 if not self.tear_down_git():
315 logging.error('Using both leaking checkout and git dirty checkout') 291 logging.error('Using both leaking checkout and git dirty checkout')
316 292
317 def tear_down(self): 293 def tear_down(self):
318 """Kills the servers and delete the directories.""" 294 """Kills the servers and delete the directories."""
319 self.tear_down_svn() 295 self.tear_down_svn()
320 self.tear_down_git() 296 self.tear_down_git()
321 if not self.SHOULD_LEAK: 297 # This deletes the directories.
322 logging.debug('Removing %s' % self.trial_dir()) 298 super(FakeReposBase, self).tear_down()
323 rmtree(self.trial_dir())
324 299
325 def tear_down_svn(self): 300 def tear_down_svn(self):
326 if self.svnserve: 301 if self.svnserve:
327 logging.debug('Killing svnserve pid %s' % self.svnserve.pid) 302 logging.debug('Killing svnserve pid %s' % self.svnserve.pid)
328 self.svnserve.kill() 303 self.svnserve.kill()
329 self.wait_for_port_to_free(self.svn_port) 304 self.wait_for_port_to_free(self.svn_port)
330 self.svnserve = None 305 self.svnserve = None
331 if not self.SHOULD_LEAK: 306 if not self.SHOULD_LEAK:
332 logging.debug('Removing %s' % self.svn_repo) 307 logging.debug('Removing %s' % self.svn_repo)
333 rmtree(self.svn_repo) 308 rmtree(self.svn_repo)
(...skipping 51 matching lines...) Expand 10 before | Expand all | Expand 10 after
385 write(join(self.svn_repo, 'conf', 'svnserve.conf'), 360 write(join(self.svn_repo, 'conf', 'svnserve.conf'),
386 '[general]\n' 361 '[general]\n'
387 'anon-access = read\n' 362 'anon-access = read\n'
388 'auth-access = write\n' 363 'auth-access = write\n'
389 'password-db = passwd\n') 364 'password-db = passwd\n')
390 text = '[users]\n' 365 text = '[users]\n'
391 text += ''.join('%s = %s\n' % (usr, pwd) for usr, pwd in self.USERS) 366 text += ''.join('%s = %s\n' % (usr, pwd) for usr, pwd in self.USERS)
392 write(join(self.svn_repo, 'conf', 'passwd'), text) 367 write(join(self.svn_repo, 'conf', 'passwd'), text)
393 368
394 # Start the daemon. 369 # Start the daemon.
395 cmd = ['svnserve', '-d', '--foreground', '-r', self.repos_dir] 370 cmd = ['svnserve', '-d', '--foreground', '-r', self.root_dir]
396 if self.host == '127.0.0.1': 371 if self.host == '127.0.0.1':
397 cmd.append('--listen-host=' + self.host) 372 cmd.append('--listen-host=' + self.host)
398 self.check_port_is_free(self.svn_port) 373 self.check_port_is_free(self.svn_port)
399 self.svnserve = Popen(cmd, cwd=self.svn_repo) 374 self.svnserve = Popen(cmd, cwd=self.svn_repo)
400 self.wait_for_port_to_bind(self.svn_port, self.svnserve) 375 self.wait_for_port_to_bind(self.svn_port, self.svnserve)
401 self.populateSvn() 376 self.populateSvn()
402 self.svn_dirty = False 377 self.svn_dirty = False
403 return True 378 return True
404 379
405 def set_up_git(self): 380 def set_up_git(self):
406 """Creates git repositories and start the servers.""" 381 """Creates git repositories and start the servers."""
407 self.set_up() 382 self.set_up()
408 if self.gitdaemon: 383 if self.gitdaemon:
409 return True 384 return True
410 if sys.platform == 'win32': 385 if sys.platform == 'win32':
411 return False 386 return False
412 assert self.git_pid_file == None 387 assert self.git_pid_file == None
413 for repo in ['repo_%d' % r for r in range(1, self.NB_GIT_REPOS + 1)]: 388 for repo in ['repo_%d' % r for r in range(1, self.NB_GIT_REPOS + 1)]:
414 check_call(['git', 'init', '-q', join(self.git_root, repo)]) 389 check_call(['git', 'init', '-q', join(self.git_root, repo)])
415 self.git_hashes[repo] = [None] 390 self.git_hashes[repo] = [None]
416 # Unlike svn, populate git before starting the server. 391 # Unlike svn, populate git before starting the server.
417 self.populateGit() 392 self.populateGit()
418 # Start the daemon. 393 # Start the daemon.
419 self.git_pid_file = tempfile.NamedTemporaryFile() 394 self.git_pid_file = tempfile.NamedTemporaryFile()
420 cmd = ['git', 'daemon', 395 cmd = ['git', 'daemon',
421 '--export-all', 396 '--export-all',
422 '--reuseaddr', 397 '--reuseaddr',
423 '--base-path=' + self.repos_dir, 398 '--base-path=' + self.root_dir,
424 '--pid-file=' + self.git_pid_file.name] 399 '--pid-file=' + self.git_pid_file.name]
425 if self.host == '127.0.0.1': 400 if self.host == '127.0.0.1':
426 cmd.append('--listen=' + self.host) 401 cmd.append('--listen=' + self.host)
427 self.check_port_is_free(self.git_port) 402 self.check_port_is_free(self.git_port)
428 self.gitdaemon = Popen(cmd, cwd=self.repos_dir) 403 self.gitdaemon = Popen(cmd, cwd=self.root_dir)
429 self.wait_for_port_to_bind(self.git_port, self.gitdaemon) 404 self.wait_for_port_to_bind(self.git_port, self.gitdaemon)
430 self.git_dirty = False 405 self.git_dirty = False
431 return True 406 return True
432 407
433 def _commit_svn(self, tree): 408 def _commit_svn(self, tree):
434 self._genTree(self.svn_checkout, tree) 409 self._genTree(self.svn_checkout, tree)
435 commit_svn(self.svn_checkout, self.USERS[0][0], self.USERS[0][1]) 410 commit_svn(self.svn_checkout, self.USERS[0][0], self.USERS[0][1])
436 if self.svn_revs and self.svn_revs[-1]: 411 if self.svn_revs and self.svn_revs[-1]:
437 new_tree = self.svn_revs[-1].copy() 412 new_tree = self.svn_revs[-1].copy()
438 new_tree.update(tree) 413 new_tree.update(tree)
(...skipping 268 matching lines...) Expand 10 before | Expand all | Expand 10 after
707 'git_base': self.git_base, 682 'git_base': self.git_base,
708 # See self.__init__() for the format. Grab's the hash of the first 683 # See self.__init__() for the format. Grab's the hash of the first
709 # commit in repo_2. Only keep the first 7 character because of: 684 # commit in repo_2. Only keep the first 7 character because of:
710 # TODO(maruel): http://crosbug.com/3591 We need to strip the hash.. duh. 685 # TODO(maruel): http://crosbug.com/3591 We need to strip the hash.. duh.
711 'hash': self.git_hashes['repo_2'][1][0][:7] 686 'hash': self.git_hashes['repo_2'][1][0][:7]
712 }, 687 },
713 'origin': 'git/repo_1@2\n', 688 'origin': 'git/repo_1@2\n',
714 }) 689 })
715 690
716 691
717 class FakeReposTestBase(unittest.TestCase): 692 class FakeReposTestBase(trial_dir.TestCase):
718 """This is vaguely inspired by twisted.""" 693 """This is vaguely inspired by twisted."""
719
720 # Replace this in your subclass.
721 CLASS_ROOT_DIR = None
722
723 # static FakeRepos instance. Lazy loaded. 694 # static FakeRepos instance. Lazy loaded.
724 FAKE_REPOS = None 695 FAKE_REPOS = None
725 # Override if necessary. 696 # Override if necessary.
726 FAKE_REPOS_CLASS = FakeRepos 697 FAKE_REPOS_CLASS = FakeRepos
727 698
728 def __init__(self, *args, **kwargs): 699 def setUp(self):
729 unittest.TestCase.__init__(self, *args, **kwargs) 700 super(FakeReposTestBase, self).setUp()
730 if not FakeReposTestBase.FAKE_REPOS: 701 if not FakeReposTestBase.FAKE_REPOS:
731 # Lazy create the global instance. 702 # Lazy create the global instance.
732 FakeReposTestBase.FAKE_REPOS = self.FAKE_REPOS_CLASS() 703 FakeReposTestBase.FAKE_REPOS = self.FAKE_REPOS_CLASS()
733 704 # No need to call self.FAKE_REPOS.setUp(), it will be called by the child
734 def setUp(self): 705 # class.
735 unittest.TestCase.setUp(self) 706 # Do not define tearDown(), since super's version does the right thing and
736 self.FAKE_REPOS.set_up() 707 # FAKE_REPOS is kept across tests.
737
738 # Remove left overs and start fresh.
739 if not self.CLASS_ROOT_DIR:
740 self.CLASS_ROOT_DIR = join(self.FAKE_REPOS.trial_dir(), 'smoke')
741 self.root_dir = join(self.CLASS_ROOT_DIR, self.id())
742 rmtree(self.root_dir)
743 os.makedirs(self.root_dir)
744
745 def tearDown(self):
746 if not self.FAKE_REPOS.SHOULD_LEAK:
747 rmtree(self.root_dir)
748 708
749 @property 709 @property
750 def svn_base(self): 710 def svn_base(self):
751 """Shortcut.""" 711 """Shortcut."""
752 return self.FAKE_REPOS.svn_base 712 return self.FAKE_REPOS.svn_base
753 713
754 @property 714 @property
755 def git_base(self): 715 def git_base(self):
756 """Shortcut.""" 716 """Shortcut."""
757 return self.FAKE_REPOS.git_base 717 return self.FAKE_REPOS.git_base
(...skipping 59 matching lines...) Expand 10 before | Expand all | Expand 10 after
817 """Sort-hand: Returns the hash for a git 'revision'.""" 777 """Sort-hand: Returns the hash for a git 'revision'."""
818 return self.FAKE_REPOS.git_hashes[repo][int(rev)][0] 778 return self.FAKE_REPOS.git_hashes[repo][int(rev)][0]
819 779
820 def gittree(self, repo, rev): 780 def gittree(self, repo, rev):
821 """Sort-hand: returns the directory tree for a git 'revision'.""" 781 """Sort-hand: returns the directory tree for a git 'revision'."""
822 return self.FAKE_REPOS.git_hashes[repo][int(rev)][1] 782 return self.FAKE_REPOS.git_hashes[repo][int(rev)][1]
823 783
824 784
825 def main(argv): 785 def main(argv):
826 fake = FakeRepos() 786 fake = FakeRepos()
827 print 'Using %s' % fake.trial_dir() 787 print 'Using %s' % fake.root_dir
828 try: 788 try:
829 fake.set_up_svn() 789 fake.set_up_svn()
830 fake.set_up_git() 790 fake.set_up_git()
831 print('Fake setup, press enter to quit or Ctrl-C to keep the checkouts.') 791 print('Fake setup, press enter to quit or Ctrl-C to keep the checkouts.')
832 sys.stdin.readline() 792 sys.stdin.readline()
833 except KeyboardInterrupt: 793 except KeyboardInterrupt:
834 fake.SHOULD_LEAK = True 794 fake.leak = True
835 return 0 795 return 0
836 796
837 797
838 if '-l' in sys.argv:
839 # See SHOULD_LEAK definition in FakeReposBase for its purpose.
840 FakeReposBase.SHOULD_LEAK = True
841 print 'Leaking!'
842 sys.argv.remove('-l')
843
844
845 if __name__ == '__main__': 798 if __name__ == '__main__':
846 sys.exit(main(sys.argv)) 799 sys.exit(main(sys.argv))
OLDNEW
« no previous file with comments | « no previous file | tests/trial_dir.py » ('j') | tests/trial_dir.py » ('J')

Powered by Google App Engine
This is Rietveld 408576698