You cannot select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
	
	
		
			345 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			Python
		
	
			
		
		
	
	
			345 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			Python
		
	
| #!/usr/bin/env python
 | |
| # Copyright (c) 2011 The Chromium Authors. All rights reserved.
 | |
| # Use of this source code is governed by a BSD-style license that can be
 | |
| # found in the LICENSE file.
 | |
| 
 | |
| """Unit tests for scm.py."""
 | |
| 
 | |
| # pylint: disable=E1101,W0403
 | |
| from __future__ import with_statement
 | |
| import logging
 | |
| import sys
 | |
| import unittest
 | |
| 
 | |
| # Fixes include path.
 | |
| from super_mox import SuperMoxTestBase
 | |
| 
 | |
| import fake_repos
 | |
| import scm
 | |
| import subprocess2
 | |
| 
 | |
| 
 | |
| class BaseTestCase(SuperMoxTestBase):
 | |
|   # Like unittest's assertRaises, but checks for Gclient.Error.
 | |
|   def assertRaisesError(self, msg, fn, *args, **kwargs):
 | |
|     try:
 | |
|       fn(*args, **kwargs)
 | |
|     except scm.gclient_utils.Error, e:
 | |
|       self.assertEquals(e.args[0], msg)
 | |
|     else:
 | |
|       self.fail('%s not raised' % msg)
 | |
| 
 | |
| 
 | |
| class BaseSCMTestCase(BaseTestCase):
 | |
|   def setUp(self):
 | |
|     BaseTestCase.setUp(self)
 | |
|     self.mox.StubOutWithMock(scm.gclient_utils, 'CheckCallAndFilter')
 | |
|     self.mox.StubOutWithMock(scm.gclient_utils, 'CheckCallAndFilterAndHeader')
 | |
|     self.mox.StubOutWithMock(subprocess2, 'Popen')
 | |
|     self.mox.StubOutWithMock(subprocess2, 'communicate')
 | |
| 
 | |
| 
 | |
| class RootTestCase(BaseSCMTestCase):
 | |
|   def testMembersChanged(self):
 | |
|     self.mox.ReplayAll()
 | |
|     members = [
 | |
|         'ElementTree', 'GetCasedPath', 'GenFakeDiff', 'GIT', 'SVN',
 | |
|         'ValidateEmail',
 | |
|         'cStringIO', 'determine_scm', 'gclient_utils', 'glob', 'logging', 'os',
 | |
|         're', 'subprocess2', 'sys', 'tempfile', 'time',
 | |
|     ]
 | |
|     # If this test fails, you should add the relevant test.
 | |
|     self.compareMembers(scm, members)
 | |
| 
 | |
| 
 | |
| class GitWrapperTestCase(BaseSCMTestCase):
 | |
|   def testMembersChanged(self):
 | |
|     members = [
 | |
|         'AssertVersion', 'Capture', 'CaptureStatus',
 | |
|         'FetchUpstreamTuple',
 | |
|         'GenerateDiff', 'GetBranch', 'GetBranchRef', 'GetCheckoutRoot',
 | |
|         'GetDifferentFiles', 'GetEmail', 'GetPatchName', 'GetSVNBranch',
 | |
|         'GetUpstreamBranch', 'IsGitSvn', 'MatchSvnGlob', 'ShortBranchName',
 | |
|     ]
 | |
|     # If this test fails, you should add the relevant test.
 | |
|     self.compareMembers(scm.GIT, members)
 | |
| 
 | |
|   def testGetEmail(self):
 | |
|     self.mox.StubOutWithMock(scm.GIT, 'Capture')
 | |
|     scm.GIT.Capture(['config', 'user.email'], cwd=self.root_dir
 | |
|                     ).AndReturn('mini@me.com')
 | |
|     self.mox.ReplayAll()
 | |
|     self.assertEqual(scm.GIT.GetEmail(self.root_dir), 'mini@me.com')
 | |
| 
 | |
|   def testMatchSvnGlob(self):
 | |
|     self.assertEquals(scm.GIT.MatchSvnGlob(
 | |
|         'svn://svn.chromium.org/chrome/trunk/src',
 | |
|         'svn://svn.chromium.org/chrome',
 | |
|         'trunk/src:refs/remotes/origin/trunk',
 | |
|         False), 'refs/remotes/origin/trunk')
 | |
|     self.assertEquals(scm.GIT.MatchSvnGlob(
 | |
|         'https://v8.googlecode.com/svn/branches/bleeding_edge',
 | |
|         'https://v8.googlecode.com/svn',
 | |
|         'branches/*:refs/remotes/*',
 | |
|         True), 'refs/remotes/bleeding_edge')
 | |
| 
 | |
| class SVNTestCase(BaseSCMTestCase):
 | |
|   def setUp(self):
 | |
|     BaseSCMTestCase.setUp(self)
 | |
|     self.mox.StubOutWithMock(scm.SVN, 'Capture')
 | |
|     self.url = self.SvnUrl()
 | |
| 
 | |
|   def testMembersChanged(self):
 | |
|     self.mox.ReplayAll()
 | |
|     members = [
 | |
|         'AssertVersion', 'Capture', 'CaptureRevision', 'CaptureInfo',
 | |
|         'CaptureStatus', 'current_version', 'DiffItem', 'GenerateDiff',
 | |
|         'GetCheckoutRoot', 'GetEmail', 'GetFileProperty', 'IsMoved',
 | |
|         'IsMovedInfo', 'ReadSimpleAuth', 'Revert', 'RunAndGetFileList',
 | |
|     ]
 | |
|     # If this test fails, you should add the relevant test.
 | |
|     self.compareMembers(scm.SVN, members)
 | |
| 
 | |
|   def testGetCheckoutRoot(self):
 | |
|     self.mox.StubOutWithMock(scm.SVN, 'CaptureInfo')
 | |
|     self.mox.StubOutWithMock(scm, 'GetCasedPath')
 | |
|     scm.os.path.abspath = lambda x: x
 | |
|     scm.GetCasedPath = lambda x: x
 | |
|     # pylint: disable=E1103
 | |
|     scm.SVN.CaptureInfo(self.root_dir + '/foo/bar').AndReturn({
 | |
|         'Repository Root': 'svn://svn.chromium.org/chrome',
 | |
|         'URL': 'svn://svn.chromium.org/chrome/trunk/src',
 | |
|     })
 | |
|     scm.SVN.CaptureInfo(self.root_dir + '/foo').AndReturn({
 | |
|         'Repository Root': 'svn://svn.chromium.org/chrome',
 | |
|         'URL': 'svn://svn.chromium.org/chrome/trunk',
 | |
|     })
 | |
|     scm.SVN.CaptureInfo(self.root_dir).AndReturn({
 | |
|         'Repository Root': 'svn://svn.chromium.org/chrome',
 | |
|         'URL': 'svn://svn.chromium.org/chrome/trunk/tools/commit-queue/workdir',
 | |
|     })
 | |
|     self.mox.ReplayAll()
 | |
|     self.assertEquals(scm.SVN.GetCheckoutRoot(self.root_dir + '/foo/bar'),
 | |
|                       self.root_dir + '/foo')
 | |
| 
 | |
|   def testGetFileInfo(self):
 | |
|     xml_text = r"""<?xml version="1.0"?>
 | |
| <info>
 | |
| <entry kind="file" path="%s" revision="14628">
 | |
| <url>http://src.chromium.org/svn/trunk/src/chrome/app/d</url>
 | |
| <repository><root>http://src.chromium.org/svn</root></repository>
 | |
| <wc-info>
 | |
| <schedule>add</schedule>
 | |
| <depth>infinity</depth>
 | |
| <copy-from-url>http://src.chromium.org/svn/trunk/src/chrome/app/DEPS</copy-from-url>
 | |
| <copy-from-rev>14628</copy-from-rev>
 | |
| <checksum>369f59057ba0e6d9017e28f8bdfb1f43</checksum>
 | |
| </wc-info>
 | |
| </entry>
 | |
| </info>
 | |
| """ % self.url
 | |
|     scm.SVN.Capture(['info', '--xml', self.url]).AndReturn(xml_text)
 | |
|     expected = {
 | |
|       'URL': 'http://src.chromium.org/svn/trunk/src/chrome/app/d',
 | |
|       'UUID': None,
 | |
|       'Repository Root': 'http://src.chromium.org/svn',
 | |
|       'Schedule': 'add',
 | |
|       'Copied From URL':
 | |
|         'http://src.chromium.org/svn/trunk/src/chrome/app/DEPS',
 | |
|       'Copied From Rev': '14628',
 | |
|       'Path': self.url,
 | |
|       'Revision': 14628,
 | |
|       'Node Kind': 'file',
 | |
|     }
 | |
|     self.mox.ReplayAll()
 | |
|     file_info = scm.SVN.CaptureInfo(self.url)
 | |
|     self.assertEquals(sorted(file_info.items()), sorted(expected.items()))
 | |
| 
 | |
|   def testCaptureInfo(self):
 | |
|     xml_text = """<?xml version="1.0"?>
 | |
| <info>
 | |
| <entry
 | |
|    kind="dir"
 | |
|    path="."
 | |
|    revision="35">
 | |
| <url>%s</url>
 | |
| <repository>
 | |
| <root>%s</root>
 | |
| <uuid>7b9385f5-0452-0410-af26-ad4892b7a1fb</uuid>
 | |
| </repository>
 | |
| <wc-info>
 | |
| <schedule>normal</schedule>
 | |
| <depth>infinity</depth>
 | |
| </wc-info>
 | |
| <commit
 | |
|    revision="35">
 | |
| <author>maruel</author>
 | |
| <date>2008-12-04T20:12:19.685120Z</date>
 | |
| </commit>
 | |
| </entry>
 | |
| </info>
 | |
| """ % (self.url, self.root_dir)
 | |
|     scm.SVN.Capture(['info', '--xml', self.url]).AndReturn(xml_text)
 | |
|     self.mox.ReplayAll()
 | |
|     file_info = scm.SVN.CaptureInfo(self.url)
 | |
|     expected = {
 | |
|       'URL': self.url,
 | |
|       'UUID': '7b9385f5-0452-0410-af26-ad4892b7a1fb',
 | |
|       'Revision': 35,
 | |
|       'Repository Root': self.root_dir,
 | |
|       'Schedule': 'normal',
 | |
|       'Copied From URL': None,
 | |
|       'Copied From Rev': None,
 | |
|       'Path': '.',
 | |
|       'Node Kind': 'directory',
 | |
|     }
 | |
|     self.assertEqual(file_info, expected)
 | |
| 
 | |
|   def testCaptureStatus(self):
 | |
|     text = r"""<?xml version="1.0"?>
 | |
| <status>
 | |
| <target path=".">
 | |
| <entry path="unversionned_file.txt">
 | |
| <wc-status props="none" item="unversioned"></wc-status>
 | |
| </entry>
 | |
| <entry path="build\internal\essential.vsprops">
 | |
| <wc-status props="normal" item="modified" revision="14628">
 | |
| <commit revision="13818">
 | |
| <author>ajwong@chromium.org</author>
 | |
| <date>2009-04-16T00:42:06.872358Z</date>
 | |
| </commit>
 | |
| </wc-status>
 | |
| </entry>
 | |
| <entry path="chrome\app\d">
 | |
| <wc-status props="none" copied="true" tree-conflicted="true" item="added">
 | |
| </wc-status>
 | |
| </entry>
 | |
| <entry path="chrome\app\DEPS">
 | |
| <wc-status props="modified" item="modified" revision="14628">
 | |
| <commit revision="1279">
 | |
| <author>brettw@google.com</author>
 | |
| <date>2008-08-23T17:16:42.090152Z</date>
 | |
| </commit>
 | |
| </wc-status>
 | |
| </entry>
 | |
| <entry path="scripts\master\factory\gclient_factory.py">
 | |
| <wc-status props="normal" item="conflicted" revision="14725">
 | |
| <commit revision="14633">
 | |
| <author>nsylvain@chromium.org</author>
 | |
| <date>2009-04-27T19:37:17.977400Z</date>
 | |
| </commit>
 | |
| </wc-status>
 | |
| </entry>
 | |
| </target>
 | |
| </status>
 | |
| """
 | |
|     scm.SVN.Capture(['status', '--xml', '.']).AndReturn(text)
 | |
| 
 | |
|     self.mox.ReplayAll()
 | |
|     info = scm.SVN.CaptureStatus('.')
 | |
|     expected = [
 | |
|       ('?      ', 'unversionned_file.txt'),
 | |
|       ('M      ', 'build\\internal\\essential.vsprops'),
 | |
|       ('A  +   ', 'chrome\\app\\d'),
 | |
|       ('MM     ', 'chrome\\app\\DEPS'),
 | |
|       ('C      ', 'scripts\\master\\factory\\gclient_factory.py'),
 | |
|     ]
 | |
|     self.assertEquals(sorted(info), sorted(expected))
 | |
| 
 | |
|   def testCaptureStatusEmpty(self):
 | |
|     text = r"""<?xml version="1.0"?>
 | |
|     <status>
 | |
|     <target
 | |
|        path="perf">
 | |
|        </target>
 | |
|        </status>"""
 | |
|     scm.SVN.Capture(['status', '--xml']).AndReturn(text)
 | |
|     self.mox.ReplayAll()
 | |
|     info = scm.SVN.CaptureStatus(None)
 | |
|     self.assertEquals(info, [])
 | |
| 
 | |
| 
 | |
| class RealSvnTest(fake_repos.FakeReposTestBase):
 | |
|   # Tests that work with a checkout.
 | |
|   def setUp(self):
 | |
|     super(RealSvnTest, self).setUp()
 | |
|     self.enabled = self.FAKE_REPOS.set_up_svn()
 | |
|     if self.enabled:
 | |
|       self.svn_root = scm.os.path.join(self.root_dir, 'base')
 | |
|       scm.SVN.Capture(
 | |
|           ['checkout', self.svn_base + 'trunk/third_party', 'base'],
 | |
|           cwd=self.root_dir)
 | |
|       self.tree = self.mangle_svn_tree(('trunk/third_party@-1', ''),)
 | |
| 
 | |
|   def _capture(self, cmd, **kwargs):
 | |
|     kwargs.setdefault('cwd', self.svn_root)
 | |
|     return scm.SVN.Capture(cmd, **kwargs)
 | |
| 
 | |
|   def testCheckout(self):
 | |
|     if not self.enabled:
 | |
|       return
 | |
|     # Checkout and verify the tree.
 | |
|     self.assertTree(self.tree, self.svn_root)
 | |
| 
 | |
|   def testRevert(self):
 | |
|     if not self.enabled:
 | |
|       return
 | |
|     # Mess around and make sure revert works for all corner cases.
 | |
|     # - svn add a file
 | |
|     # - svn add a file and delete it
 | |
|     # - Delete a file
 | |
|     # - svn delete a file
 | |
|     # - svn move a directory and svn rename files in it
 | |
|     # - add a directory tree.
 | |
|     def join(*args):
 | |
|       return scm.os.path.join(self.svn_root, *args)
 | |
|     self._capture(['move', 'foo', 'foo2'])
 | |
|     self._capture(
 | |
|         ['move',
 | |
|          scm.os.path.join('foo2', 'origin'),
 | |
|          scm.os.path.join('foo2', 'o')])
 | |
|     scm.os.remove(join('origin'))
 | |
|     self._capture(['propset', 'foo', 'bar', join('prout', 'origin')])
 | |
|     fake_repos.gclient_utils.rmtree(join('prout'))
 | |
|     with open(join('faa'), 'w') as f:
 | |
|       f.write('eh')
 | |
|     with open(join('faala'), 'w') as f:
 | |
|       f.write('oh')
 | |
|     self._capture(['add', join('faala')])
 | |
|     added_and_removed = join('added_and_removed')
 | |
|     with open(added_and_removed, 'w') as f:
 | |
|       f.write('oh')
 | |
|     self._capture(['add', added_and_removed])
 | |
|     scm.os.remove(added_and_removed)
 | |
|     # Make sure a tree of directories can be removed.
 | |
|     scm.os.makedirs(join('new_dir', 'subdir'))
 | |
|     with open(join('new_dir', 'subdir', 'newfile'), 'w') as f:
 | |
|       f.write('ah!')
 | |
|     self._capture(['add', join('new_dir')])
 | |
|     self._capture(['add', join('new_dir', 'subdir')])
 | |
|     self._capture(['add', join('new_dir', 'subdir', 'newfile')])
 | |
|     # A random file in an added directory confuses svn.
 | |
|     scm.os.makedirs(join('new_dir2', 'subdir'))
 | |
|     with open(join('new_dir2', 'subdir', 'newfile'), 'w') as f:
 | |
|       f.write('ah!')
 | |
|     self._capture(['add', join('new_dir2')])
 | |
|     self._capture(['add', join('new_dir2', 'subdir')])
 | |
|     self._capture(['add', join('new_dir2', 'subdir', 'newfile')])
 | |
|     with open(join('new_dir2', 'subdir', 'unversionedfile'), 'w') as f:
 | |
|       f.write('unadded file!')
 | |
| 
 | |
|     scm.SVN.Revert(self.svn_root)
 | |
|     self._capture(['update', '--revision', 'base'])
 | |
| 
 | |
|     self.assertTree(self.tree, self.svn_root)
 | |
|     # Asserting the tree is not sufficient, svn status must come out clear too.
 | |
|     self.assertEquals('', self._capture(['status']))
 | |
| 
 | |
| 
 | |
| if __name__ == '__main__':
 | |
|   if '-v' in sys.argv:
 | |
|     logging.basicConfig(level=logging.DEBUG)
 | |
|   unittest.main()
 | |
| 
 | |
| # vim: ts=2:sw=2:tw=80:et:
 |