test_wrapper.py 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509
  1. # -*- coding:utf-8 -*-
  2. #
  3. # Copyright (C) 2015 The Android Open Source Project
  4. #
  5. # Licensed under the Apache License, Version 2.0 (the "License");
  6. # you may not use this file except in compliance with the License.
  7. # You may obtain a copy of the License at
  8. #
  9. # http://www.apache.org/licenses/LICENSE-2.0
  10. #
  11. # Unless required by applicable law or agreed to in writing, software
  12. # distributed under the License is distributed on an "AS IS" BASIS,
  13. # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  14. # See the License for the specific language governing permissions and
  15. # limitations under the License.
  16. """Unittests for the wrapper.py module."""
  17. from __future__ import print_function
  18. import contextlib
  19. import os
  20. import re
  21. import shutil
  22. import tempfile
  23. import unittest
  24. import git_command
  25. import main
  26. import platform_utils
  27. from pyversion import is_python3
  28. import wrapper
  29. if is_python3():
  30. from unittest import mock
  31. from io import StringIO
  32. else:
  33. import mock
  34. from StringIO import StringIO
  35. @contextlib.contextmanager
  36. def TemporaryDirectory():
  37. """Create a new empty git checkout for testing."""
  38. # TODO(vapier): Convert this to tempfile.TemporaryDirectory once we drop
  39. # Python 2 support entirely.
  40. try:
  41. tempdir = tempfile.mkdtemp(prefix='repo-tests')
  42. yield tempdir
  43. finally:
  44. platform_utils.rmtree(tempdir)
  45. def fixture(*paths):
  46. """Return a path relative to tests/fixtures.
  47. """
  48. return os.path.join(os.path.dirname(__file__), 'fixtures', *paths)
  49. class RepoWrapperTestCase(unittest.TestCase):
  50. """TestCase for the wrapper module."""
  51. def setUp(self):
  52. """Load the wrapper module every time."""
  53. wrapper._wrapper_module = None
  54. self.wrapper = wrapper.Wrapper()
  55. if not is_python3():
  56. self.assertRegex = self.assertRegexpMatches
  57. class RepoWrapperUnitTest(RepoWrapperTestCase):
  58. """Tests helper functions in the repo wrapper
  59. """
  60. def test_version(self):
  61. """Make sure _Version works."""
  62. with self.assertRaises(SystemExit) as e:
  63. with mock.patch('sys.stdout', new_callable=StringIO) as stdout:
  64. with mock.patch('sys.stderr', new_callable=StringIO) as stderr:
  65. self.wrapper._Version()
  66. self.assertEqual(0, e.exception.code)
  67. self.assertEqual('', stderr.getvalue())
  68. self.assertIn('repo launcher version', stdout.getvalue())
  69. def test_python_constraints(self):
  70. """The launcher should never require newer than main.py."""
  71. self.assertGreaterEqual(main.MIN_PYTHON_VERSION_HARD,
  72. wrapper.MIN_PYTHON_VERSION_HARD)
  73. self.assertGreaterEqual(main.MIN_PYTHON_VERSION_SOFT,
  74. wrapper.MIN_PYTHON_VERSION_SOFT)
  75. # Make sure the versions are themselves in sync.
  76. self.assertGreaterEqual(wrapper.MIN_PYTHON_VERSION_SOFT,
  77. wrapper.MIN_PYTHON_VERSION_HARD)
  78. def test_init_parser(self):
  79. """Make sure 'init' GetParser works."""
  80. parser = self.wrapper.GetParser(gitc_init=False)
  81. opts, args = parser.parse_args([])
  82. self.assertEqual([], args)
  83. self.assertIsNone(opts.manifest_url)
  84. def test_gitc_init_parser(self):
  85. """Make sure 'gitc-init' GetParser works."""
  86. parser = self.wrapper.GetParser(gitc_init=True)
  87. opts, args = parser.parse_args([])
  88. self.assertEqual([], args)
  89. self.assertIsNone(opts.manifest_file)
  90. def test_get_gitc_manifest_dir_no_gitc(self):
  91. """
  92. Test reading a missing gitc config file
  93. """
  94. self.wrapper.GITC_CONFIG_FILE = fixture('missing_gitc_config')
  95. val = self.wrapper.get_gitc_manifest_dir()
  96. self.assertEqual(val, '')
  97. def test_get_gitc_manifest_dir(self):
  98. """
  99. Test reading the gitc config file and parsing the directory
  100. """
  101. self.wrapper.GITC_CONFIG_FILE = fixture('gitc_config')
  102. val = self.wrapper.get_gitc_manifest_dir()
  103. self.assertEqual(val, '/test/usr/local/google/gitc')
  104. def test_gitc_parse_clientdir_no_gitc(self):
  105. """
  106. Test parsing the gitc clientdir without gitc running
  107. """
  108. self.wrapper.GITC_CONFIG_FILE = fixture('missing_gitc_config')
  109. self.assertEqual(self.wrapper.gitc_parse_clientdir('/something'), None)
  110. self.assertEqual(self.wrapper.gitc_parse_clientdir('/gitc/manifest-rw/test'), 'test')
  111. def test_gitc_parse_clientdir(self):
  112. """
  113. Test parsing the gitc clientdir
  114. """
  115. self.wrapper.GITC_CONFIG_FILE = fixture('gitc_config')
  116. self.assertEqual(self.wrapper.gitc_parse_clientdir('/something'), None)
  117. self.assertEqual(self.wrapper.gitc_parse_clientdir('/gitc/manifest-rw/test'), 'test')
  118. self.assertEqual(self.wrapper.gitc_parse_clientdir('/gitc/manifest-rw/test/'), 'test')
  119. self.assertEqual(self.wrapper.gitc_parse_clientdir('/gitc/manifest-rw/test/extra'), 'test')
  120. self.assertEqual(self.wrapper.gitc_parse_clientdir('/test/usr/local/google/gitc/test'), 'test')
  121. self.assertEqual(self.wrapper.gitc_parse_clientdir('/test/usr/local/google/gitc/test/'), 'test')
  122. self.assertEqual(self.wrapper.gitc_parse_clientdir('/test/usr/local/google/gitc/test/extra'),
  123. 'test')
  124. self.assertEqual(self.wrapper.gitc_parse_clientdir('/gitc/manifest-rw/'), None)
  125. self.assertEqual(self.wrapper.gitc_parse_clientdir('/test/usr/local/google/gitc/'), None)
  126. class SetGitTrace2ParentSid(RepoWrapperTestCase):
  127. """Check SetGitTrace2ParentSid behavior."""
  128. KEY = 'GIT_TRACE2_PARENT_SID'
  129. VALID_FORMAT = re.compile(r'^repo-[0-9]{8}T[0-9]{6}Z-P[0-9a-f]{8}$')
  130. def test_first_set(self):
  131. """Test env var not yet set."""
  132. env = {}
  133. self.wrapper.SetGitTrace2ParentSid(env)
  134. self.assertIn(self.KEY, env)
  135. value = env[self.KEY]
  136. self.assertRegex(value, self.VALID_FORMAT)
  137. def test_append(self):
  138. """Test env var is appended."""
  139. env = {self.KEY: 'pfx'}
  140. self.wrapper.SetGitTrace2ParentSid(env)
  141. self.assertIn(self.KEY, env)
  142. value = env[self.KEY]
  143. self.assertTrue(value.startswith('pfx/'))
  144. self.assertRegex(value[4:], self.VALID_FORMAT)
  145. def test_global_context(self):
  146. """Check os.environ gets updated by default."""
  147. os.environ.pop(self.KEY, None)
  148. self.wrapper.SetGitTrace2ParentSid()
  149. self.assertIn(self.KEY, os.environ)
  150. value = os.environ[self.KEY]
  151. self.assertRegex(value, self.VALID_FORMAT)
  152. class RunCommand(RepoWrapperTestCase):
  153. """Check run_command behavior."""
  154. def test_capture(self):
  155. """Check capture_output handling."""
  156. ret = self.wrapper.run_command(['echo', 'hi'], capture_output=True)
  157. self.assertEqual(ret.stdout, 'hi\n')
  158. def test_check(self):
  159. """Check check handling."""
  160. self.wrapper.run_command(['true'], check=False)
  161. self.wrapper.run_command(['true'], check=True)
  162. self.wrapper.run_command(['false'], check=False)
  163. with self.assertRaises(self.wrapper.RunError):
  164. self.wrapper.run_command(['false'], check=True)
  165. class RunGit(RepoWrapperTestCase):
  166. """Check run_git behavior."""
  167. def test_capture(self):
  168. """Check capture_output handling."""
  169. ret = self.wrapper.run_git('--version')
  170. self.assertIn('git', ret.stdout)
  171. def test_check(self):
  172. """Check check handling."""
  173. with self.assertRaises(self.wrapper.CloneFailure):
  174. self.wrapper.run_git('--version-asdfasdf')
  175. self.wrapper.run_git('--version-asdfasdf', check=False)
  176. class ParseGitVersion(RepoWrapperTestCase):
  177. """Check ParseGitVersion behavior."""
  178. def test_autoload(self):
  179. """Check we can load the version from the live git."""
  180. ret = self.wrapper.ParseGitVersion()
  181. self.assertIsNotNone(ret)
  182. def test_bad_ver(self):
  183. """Check handling of bad git versions."""
  184. ret = self.wrapper.ParseGitVersion(ver_str='asdf')
  185. self.assertIsNone(ret)
  186. def test_normal_ver(self):
  187. """Check handling of normal git versions."""
  188. ret = self.wrapper.ParseGitVersion(ver_str='git version 2.25.1')
  189. self.assertEqual(2, ret.major)
  190. self.assertEqual(25, ret.minor)
  191. self.assertEqual(1, ret.micro)
  192. self.assertEqual('2.25.1', ret.full)
  193. def test_extended_ver(self):
  194. """Check handling of extended distro git versions."""
  195. ret = self.wrapper.ParseGitVersion(
  196. ver_str='git version 1.30.50.696.g5e7596f4ac-goog')
  197. self.assertEqual(1, ret.major)
  198. self.assertEqual(30, ret.minor)
  199. self.assertEqual(50, ret.micro)
  200. self.assertEqual('1.30.50.696.g5e7596f4ac-goog', ret.full)
  201. class CheckGitVersion(RepoWrapperTestCase):
  202. """Check _CheckGitVersion behavior."""
  203. def test_unknown(self):
  204. """Unknown versions should abort."""
  205. with mock.patch.object(self.wrapper, 'ParseGitVersion', return_value=None):
  206. with self.assertRaises(self.wrapper.CloneFailure):
  207. self.wrapper._CheckGitVersion()
  208. def test_old(self):
  209. """Old versions should abort."""
  210. with mock.patch.object(
  211. self.wrapper, 'ParseGitVersion',
  212. return_value=self.wrapper.GitVersion(1, 0, 0, '1.0.0')):
  213. with self.assertRaises(self.wrapper.CloneFailure):
  214. self.wrapper._CheckGitVersion()
  215. def test_new(self):
  216. """Newer versions should run fine."""
  217. with mock.patch.object(
  218. self.wrapper, 'ParseGitVersion',
  219. return_value=self.wrapper.GitVersion(100, 0, 0, '100.0.0')):
  220. self.wrapper._CheckGitVersion()
  221. class NeedSetupGnuPG(RepoWrapperTestCase):
  222. """Check NeedSetupGnuPG behavior."""
  223. def test_missing_dir(self):
  224. """The ~/.repoconfig tree doesn't exist yet."""
  225. with TemporaryDirectory() as tempdir:
  226. self.wrapper.home_dot_repo = os.path.join(tempdir, 'foo')
  227. self.assertTrue(self.wrapper.NeedSetupGnuPG())
  228. def test_missing_keyring(self):
  229. """The keyring-version file doesn't exist yet."""
  230. with TemporaryDirectory() as tempdir:
  231. self.wrapper.home_dot_repo = tempdir
  232. self.assertTrue(self.wrapper.NeedSetupGnuPG())
  233. def test_empty_keyring(self):
  234. """The keyring-version file exists, but is empty."""
  235. with TemporaryDirectory() as tempdir:
  236. self.wrapper.home_dot_repo = tempdir
  237. with open(os.path.join(tempdir, 'keyring-version'), 'w'):
  238. pass
  239. self.assertTrue(self.wrapper.NeedSetupGnuPG())
  240. def test_old_keyring(self):
  241. """The keyring-version file exists, but it's old."""
  242. with TemporaryDirectory() as tempdir:
  243. self.wrapper.home_dot_repo = tempdir
  244. with open(os.path.join(tempdir, 'keyring-version'), 'w') as fp:
  245. fp.write('1.0\n')
  246. self.assertTrue(self.wrapper.NeedSetupGnuPG())
  247. def test_new_keyring(self):
  248. """The keyring-version file exists, and is up-to-date."""
  249. with TemporaryDirectory() as tempdir:
  250. self.wrapper.home_dot_repo = tempdir
  251. with open(os.path.join(tempdir, 'keyring-version'), 'w') as fp:
  252. fp.write('1000.0\n')
  253. self.assertFalse(self.wrapper.NeedSetupGnuPG())
  254. class SetupGnuPG(RepoWrapperTestCase):
  255. """Check SetupGnuPG behavior."""
  256. def test_full(self):
  257. """Make sure it works completely."""
  258. with TemporaryDirectory() as tempdir:
  259. self.wrapper.home_dot_repo = tempdir
  260. self.wrapper.gpg_dir = os.path.join(self.wrapper.home_dot_repo, 'gnupg')
  261. self.assertTrue(self.wrapper.SetupGnuPG(True))
  262. with open(os.path.join(tempdir, 'keyring-version'), 'r') as fp:
  263. data = fp.read()
  264. self.assertEqual('.'.join(str(x) for x in self.wrapper.KEYRING_VERSION),
  265. data.strip())
  266. class VerifyRev(RepoWrapperTestCase):
  267. """Check verify_rev behavior."""
  268. def test_verify_passes(self):
  269. """Check when we have a valid signed tag."""
  270. desc_result = self.wrapper.RunResult(0, 'v1.0\n', '')
  271. gpg_result = self.wrapper.RunResult(0, '', '')
  272. with mock.patch.object(self.wrapper, 'run_git',
  273. side_effect=(desc_result, gpg_result)):
  274. ret = self.wrapper.verify_rev('/', 'refs/heads/stable', '1234', True)
  275. self.assertEqual('v1.0^0', ret)
  276. def test_unsigned_commit(self):
  277. """Check we fall back to signed tag when we have an unsigned commit."""
  278. desc_result = self.wrapper.RunResult(0, 'v1.0-10-g1234\n', '')
  279. gpg_result = self.wrapper.RunResult(0, '', '')
  280. with mock.patch.object(self.wrapper, 'run_git',
  281. side_effect=(desc_result, gpg_result)):
  282. ret = self.wrapper.verify_rev('/', 'refs/heads/stable', '1234', True)
  283. self.assertEqual('v1.0^0', ret)
  284. def test_verify_fails(self):
  285. """Check we fall back to signed tag when we have an unsigned commit."""
  286. desc_result = self.wrapper.RunResult(0, 'v1.0-10-g1234\n', '')
  287. gpg_result = Exception
  288. with mock.patch.object(self.wrapper, 'run_git',
  289. side_effect=(desc_result, gpg_result)):
  290. with self.assertRaises(Exception):
  291. self.wrapper.verify_rev('/', 'refs/heads/stable', '1234', True)
  292. class GitCheckoutTestCase(RepoWrapperTestCase):
  293. """Tests that use a real/small git checkout."""
  294. GIT_DIR = None
  295. REV_LIST = None
  296. @classmethod
  297. def setUpClass(cls):
  298. # Create a repo to operate on, but do it once per-class.
  299. cls.GIT_DIR = tempfile.mkdtemp(prefix='repo-rev-tests')
  300. run_git = wrapper.Wrapper().run_git
  301. remote = os.path.join(cls.GIT_DIR, 'remote')
  302. os.mkdir(remote)
  303. # Tests need to assume, that main is default branch at init,
  304. # which is not supported in config until 2.28.
  305. if git_command.git_require((2, 28, 0)):
  306. initstr = '--initial-branch=main'
  307. else:
  308. # Use template dir for init.
  309. templatedir = tempfile.mkdtemp(prefix='.test-template')
  310. with open(os.path.join(templatedir, 'HEAD'), 'w') as fp:
  311. fp.write('ref: refs/heads/main\n')
  312. initstr = '--template=' + templatedir
  313. run_git('init', initstr, cwd=remote)
  314. run_git('commit', '--allow-empty', '-minit', cwd=remote)
  315. run_git('branch', 'stable', cwd=remote)
  316. run_git('tag', 'v1.0', cwd=remote)
  317. run_git('commit', '--allow-empty', '-m2nd commit', cwd=remote)
  318. cls.REV_LIST = run_git('rev-list', 'HEAD', cwd=remote).stdout.splitlines()
  319. run_git('init', cwd=cls.GIT_DIR)
  320. run_git('fetch', remote, '+refs/heads/*:refs/remotes/origin/*', cwd=cls.GIT_DIR)
  321. @classmethod
  322. def tearDownClass(cls):
  323. if not cls.GIT_DIR:
  324. return
  325. shutil.rmtree(cls.GIT_DIR)
  326. class ResolveRepoRev(GitCheckoutTestCase):
  327. """Check resolve_repo_rev behavior."""
  328. def test_explicit_branch(self):
  329. """Check refs/heads/branch argument."""
  330. rrev, lrev = self.wrapper.resolve_repo_rev(self.GIT_DIR, 'refs/heads/stable')
  331. self.assertEqual('refs/heads/stable', rrev)
  332. self.assertEqual(self.REV_LIST[1], lrev)
  333. with self.assertRaises(wrapper.CloneFailure):
  334. self.wrapper.resolve_repo_rev(self.GIT_DIR, 'refs/heads/unknown')
  335. def test_explicit_tag(self):
  336. """Check refs/tags/tag argument."""
  337. rrev, lrev = self.wrapper.resolve_repo_rev(self.GIT_DIR, 'refs/tags/v1.0')
  338. self.assertEqual('refs/tags/v1.0', rrev)
  339. self.assertEqual(self.REV_LIST[1], lrev)
  340. with self.assertRaises(wrapper.CloneFailure):
  341. self.wrapper.resolve_repo_rev(self.GIT_DIR, 'refs/tags/unknown')
  342. def test_branch_name(self):
  343. """Check branch argument."""
  344. rrev, lrev = self.wrapper.resolve_repo_rev(self.GIT_DIR, 'stable')
  345. self.assertEqual('refs/heads/stable', rrev)
  346. self.assertEqual(self.REV_LIST[1], lrev)
  347. rrev, lrev = self.wrapper.resolve_repo_rev(self.GIT_DIR, 'main')
  348. self.assertEqual('refs/heads/main', rrev)
  349. self.assertEqual(self.REV_LIST[0], lrev)
  350. def test_tag_name(self):
  351. """Check tag argument."""
  352. rrev, lrev = self.wrapper.resolve_repo_rev(self.GIT_DIR, 'v1.0')
  353. self.assertEqual('refs/tags/v1.0', rrev)
  354. self.assertEqual(self.REV_LIST[1], lrev)
  355. def test_full_commit(self):
  356. """Check specific commit argument."""
  357. commit = self.REV_LIST[0]
  358. rrev, lrev = self.wrapper.resolve_repo_rev(self.GIT_DIR, commit)
  359. self.assertEqual(commit, rrev)
  360. self.assertEqual(commit, lrev)
  361. def test_partial_commit(self):
  362. """Check specific (partial) commit argument."""
  363. commit = self.REV_LIST[0][0:20]
  364. rrev, lrev = self.wrapper.resolve_repo_rev(self.GIT_DIR, commit)
  365. self.assertEqual(self.REV_LIST[0], rrev)
  366. self.assertEqual(self.REV_LIST[0], lrev)
  367. def test_unknown(self):
  368. """Check unknown ref/commit argument."""
  369. with self.assertRaises(wrapper.CloneFailure):
  370. self.wrapper.resolve_repo_rev(self.GIT_DIR, 'boooooooya')
  371. class CheckRepoVerify(RepoWrapperTestCase):
  372. """Check check_repo_verify behavior."""
  373. def test_no_verify(self):
  374. """Always fail with --no-repo-verify."""
  375. self.assertFalse(self.wrapper.check_repo_verify(False))
  376. def test_gpg_initialized(self):
  377. """Should pass if gpg is setup already."""
  378. with mock.patch.object(self.wrapper, 'NeedSetupGnuPG', return_value=False):
  379. self.assertTrue(self.wrapper.check_repo_verify(True))
  380. def test_need_gpg_setup(self):
  381. """Should pass/fail based on gpg setup."""
  382. with mock.patch.object(self.wrapper, 'NeedSetupGnuPG', return_value=True):
  383. with mock.patch.object(self.wrapper, 'SetupGnuPG') as m:
  384. m.return_value = True
  385. self.assertTrue(self.wrapper.check_repo_verify(True))
  386. m.return_value = False
  387. self.assertFalse(self.wrapper.check_repo_verify(True))
  388. class CheckRepoRev(GitCheckoutTestCase):
  389. """Check check_repo_rev behavior."""
  390. def test_verify_works(self):
  391. """Should pass when verification passes."""
  392. with mock.patch.object(self.wrapper, 'check_repo_verify', return_value=True):
  393. with mock.patch.object(self.wrapper, 'verify_rev', return_value='12345'):
  394. rrev, lrev = self.wrapper.check_repo_rev(self.GIT_DIR, 'stable')
  395. self.assertEqual('refs/heads/stable', rrev)
  396. self.assertEqual('12345', lrev)
  397. def test_verify_fails(self):
  398. """Should fail when verification fails."""
  399. with mock.patch.object(self.wrapper, 'check_repo_verify', return_value=True):
  400. with mock.patch.object(self.wrapper, 'verify_rev', side_effect=Exception):
  401. with self.assertRaises(Exception):
  402. self.wrapper.check_repo_rev(self.GIT_DIR, 'stable')
  403. def test_verify_ignore(self):
  404. """Should pass when verification is disabled."""
  405. with mock.patch.object(self.wrapper, 'verify_rev', side_effect=Exception):
  406. rrev, lrev = self.wrapper.check_repo_rev(self.GIT_DIR, 'stable', repo_verify=False)
  407. self.assertEqual('refs/heads/stable', rrev)
  408. self.assertEqual(self.REV_LIST[1], lrev)
  409. if __name__ == '__main__':
  410. unittest.main()