test_client.py 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532
  1. # test_client.py -- Compatibilty tests for git client.
  2. # Copyright (C) 2010 Google, Inc.
  3. #
  4. # This program is free software; you can redistribute it and/or
  5. # modify it under the terms of the GNU General Public License
  6. # as published by the Free Software Foundation; version 2
  7. # of the License or (at your option) any later version of
  8. # the License.
  9. #
  10. # This program is distributed in the hope that it will be useful,
  11. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  12. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  13. # GNU General Public License for more details.
  14. #
  15. # You should have received a copy of the GNU General Public License
  16. # along with this program; if not, write to the Free Software
  17. # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston,
  18. # MA 02110-1301, USA.
  19. """Compatibilty tests between the Dulwich client and the cgit server."""
  20. from contextlib import closing
  21. import copy
  22. from io import BytesIO
  23. import os
  24. import select
  25. import signal
  26. import subprocess
  27. import sys
  28. import tarfile
  29. import tempfile
  30. import threading
  31. try:
  32. from urlparse import unquote
  33. except ImportError:
  34. from urllib.parse import unquote
  35. try:
  36. import BaseHTTPServer
  37. import SimpleHTTPServer
  38. except ImportError:
  39. import http.server
  40. BaseHTTPServer = http.server
  41. SimpleHTTPServer = http.server
  42. if sys.platform == 'win32':
  43. import ctypes
  44. from dulwich import (
  45. client,
  46. errors,
  47. file,
  48. index,
  49. protocol,
  50. objects,
  51. repo,
  52. )
  53. from dulwich.tests import (
  54. get_safe_env,
  55. SkipTest,
  56. expectedFailure,
  57. )
  58. from dulwich.tests.utils import (
  59. skipIfPY3,
  60. )
  61. from dulwich.tests.compat.utils import (
  62. CompatTestCase,
  63. check_for_daemon,
  64. import_repo_to_dir,
  65. run_git_or_fail,
  66. _DEFAULT_GIT,
  67. rmtree_ro,
  68. )
  69. class DulwichClientTestBase(object):
  70. """Tests for client/server compatibility."""
  71. def setUp(self):
  72. self.gitroot = os.path.dirname(import_repo_to_dir('server_new.export').rstrip(os.sep))
  73. self.dest = os.path.join(self.gitroot, 'dest')
  74. file.ensure_dir_exists(self.dest)
  75. run_git_or_fail(['init', '--quiet', '--bare'], cwd=self.dest)
  76. def tearDown(self):
  77. rmtree_ro(self.gitroot)
  78. def assertDestEqualsSrc(self):
  79. repo_dir = os.path.join(self.gitroot, 'server_new.export')
  80. dest_repo_dir = os.path.join(self.gitroot, 'dest')
  81. with closing(repo.Repo(repo_dir)) as src:
  82. with closing(repo.Repo(dest_repo_dir)) as dest:
  83. self.assertReposEqual(src, dest)
  84. def _client(self):
  85. raise NotImplementedError()
  86. def _build_path(self):
  87. raise NotImplementedError()
  88. def _do_send_pack(self):
  89. c = self._client()
  90. srcpath = os.path.join(self.gitroot, 'server_new.export')
  91. with closing(repo.Repo(srcpath)) as src:
  92. sendrefs = dict(src.get_refs())
  93. del sendrefs[b'HEAD']
  94. c.send_pack(self._build_path('/dest'), lambda _: sendrefs,
  95. src.object_store.generate_pack_contents)
  96. def test_send_pack(self):
  97. self._do_send_pack()
  98. self.assertDestEqualsSrc()
  99. def test_send_pack_nothing_to_send(self):
  100. self._do_send_pack()
  101. self.assertDestEqualsSrc()
  102. # nothing to send, but shouldn't raise either.
  103. self._do_send_pack()
  104. def test_send_without_report_status(self):
  105. c = self._client()
  106. c._send_capabilities.remove(b'report-status')
  107. srcpath = os.path.join(self.gitroot, 'server_new.export')
  108. with closing(repo.Repo(srcpath)) as src:
  109. sendrefs = dict(src.get_refs())
  110. del sendrefs[b'HEAD']
  111. c.send_pack(self._build_path('/dest'), lambda _: sendrefs,
  112. src.object_store.generate_pack_contents)
  113. self.assertDestEqualsSrc()
  114. def make_dummy_commit(self, dest):
  115. b = objects.Blob.from_string(b'hi')
  116. dest.object_store.add_object(b)
  117. t = index.commit_tree(dest.object_store, [(b'hi', b.id, 0o100644)])
  118. c = objects.Commit()
  119. c.author = c.committer = b'Foo Bar <foo@example.com>'
  120. c.author_time = c.commit_time = 0
  121. c.author_timezone = c.commit_timezone = 0
  122. c.message = b'hi'
  123. c.tree = t
  124. dest.object_store.add_object(c)
  125. return c.id
  126. def disable_ff_and_make_dummy_commit(self):
  127. # disable non-fast-forward pushes to the server
  128. dest = repo.Repo(os.path.join(self.gitroot, 'dest'))
  129. run_git_or_fail(['config', 'receive.denyNonFastForwards', 'true'],
  130. cwd=dest.path)
  131. commit_id = self.make_dummy_commit(dest)
  132. return dest, commit_id
  133. def compute_send(self, src):
  134. sendrefs = dict(src.get_refs())
  135. del sendrefs[b'HEAD']
  136. return sendrefs, src.object_store.generate_pack_contents
  137. def test_send_pack_one_error(self):
  138. dest, dummy_commit = self.disable_ff_and_make_dummy_commit()
  139. dest.refs[b'refs/heads/master'] = dummy_commit
  140. repo_dir = os.path.join(self.gitroot, 'server_new.export')
  141. with closing(repo.Repo(repo_dir)) as src:
  142. sendrefs, gen_pack = self.compute_send(src)
  143. c = self._client()
  144. try:
  145. c.send_pack(self._build_path('/dest'), lambda _: sendrefs, gen_pack)
  146. except errors.UpdateRefsError as e:
  147. self.assertEqual('refs/heads/master failed to update',
  148. e.args[0])
  149. self.assertEqual({b'refs/heads/branch': b'ok',
  150. b'refs/heads/master': b'non-fast-forward'},
  151. e.ref_status)
  152. def test_send_pack_multiple_errors(self):
  153. dest, dummy = self.disable_ff_and_make_dummy_commit()
  154. # set up for two non-ff errors
  155. branch, master = b'refs/heads/branch', b'refs/heads/master'
  156. dest.refs[branch] = dest.refs[master] = dummy
  157. repo_dir = os.path.join(self.gitroot, 'server_new.export')
  158. with closing(repo.Repo(repo_dir)) as src:
  159. sendrefs, gen_pack = self.compute_send(src)
  160. c = self._client()
  161. try:
  162. c.send_pack(self._build_path('/dest'), lambda _: sendrefs, gen_pack)
  163. except errors.UpdateRefsError as e:
  164. self.assertIn(str(e),
  165. ['{0}, {1} failed to update'.format(
  166. branch.decode('ascii'), master.decode('ascii')),
  167. '{1}, {0} failed to update'.format(
  168. branch.decode('ascii'), master.decode('ascii'))])
  169. self.assertEqual({branch: b'non-fast-forward',
  170. master: b'non-fast-forward'},
  171. e.ref_status)
  172. def test_archive(self):
  173. c = self._client()
  174. f = BytesIO()
  175. c.archive(self._build_path('/server_new.export'), b'HEAD', f.write)
  176. f.seek(0)
  177. tf = tarfile.open(fileobj=f)
  178. self.assertEqual(['baz', 'foo'], tf.getnames())
  179. def test_fetch_pack(self):
  180. c = self._client()
  181. with closing(repo.Repo(os.path.join(self.gitroot, 'dest'))) as dest:
  182. refs = c.fetch(self._build_path('/server_new.export'), dest)
  183. for r in refs.items():
  184. dest.refs.set_if_equals(r[0], None, r[1])
  185. self.assertDestEqualsSrc()
  186. def test_incremental_fetch_pack(self):
  187. self.test_fetch_pack()
  188. dest, dummy = self.disable_ff_and_make_dummy_commit()
  189. dest.refs[b'refs/heads/master'] = dummy
  190. c = self._client()
  191. repo_dir = os.path.join(self.gitroot, 'server_new.export')
  192. with closing(repo.Repo(repo_dir)) as dest:
  193. refs = c.fetch(self._build_path('/dest'), dest)
  194. for r in refs.items():
  195. dest.refs.set_if_equals(r[0], None, r[1])
  196. self.assertDestEqualsSrc()
  197. def test_fetch_pack_no_side_band_64k(self):
  198. c = self._client()
  199. c._fetch_capabilities.remove(b'side-band-64k')
  200. with closing(repo.Repo(os.path.join(self.gitroot, 'dest'))) as dest:
  201. refs = c.fetch(self._build_path('/server_new.export'), dest)
  202. for r in refs.items():
  203. dest.refs.set_if_equals(r[0], None, r[1])
  204. self.assertDestEqualsSrc()
  205. def test_fetch_pack_zero_sha(self):
  206. # zero sha1s are already present on the client, and should
  207. # be ignored
  208. c = self._client()
  209. with closing(repo.Repo(os.path.join(self.gitroot, 'dest'))) as dest:
  210. refs = c.fetch(self._build_path('/server_new.export'), dest,
  211. lambda refs: [protocol.ZERO_SHA])
  212. for r in refs.items():
  213. dest.refs.set_if_equals(r[0], None, r[1])
  214. def test_send_remove_branch(self):
  215. with closing(repo.Repo(os.path.join(self.gitroot, 'dest'))) as dest:
  216. dummy_commit = self.make_dummy_commit(dest)
  217. dest.refs[b'refs/heads/master'] = dummy_commit
  218. dest.refs[b'refs/heads/abranch'] = dummy_commit
  219. sendrefs = dict(dest.refs)
  220. sendrefs[b'refs/heads/abranch'] = b"00" * 20
  221. del sendrefs[b'HEAD']
  222. gen_pack = lambda have, want: []
  223. c = self._client()
  224. self.assertEqual(dest.refs[b"refs/heads/abranch"], dummy_commit)
  225. c.send_pack(self._build_path('/dest'), lambda _: sendrefs, gen_pack)
  226. self.assertFalse(b"refs/heads/abranch" in dest.refs)
  227. def test_get_refs(self):
  228. c = self._client()
  229. refs = c.get_refs(self._build_path('/server_new.export'))
  230. repo_dir = os.path.join(self.gitroot, 'server_new.export')
  231. with closing(repo.Repo(repo_dir)) as dest:
  232. self.assertDictEqual(dest.refs.as_dict(), refs)
  233. class DulwichTCPClientTest(CompatTestCase, DulwichClientTestBase):
  234. def setUp(self):
  235. CompatTestCase.setUp(self)
  236. DulwichClientTestBase.setUp(self)
  237. if check_for_daemon(limit=1):
  238. raise SkipTest('git-daemon was already running on port %s' %
  239. protocol.TCP_GIT_PORT)
  240. env = get_safe_env()
  241. fd, self.pidfile = tempfile.mkstemp(prefix='dulwich-test-git-client',
  242. suffix=".pid")
  243. os.fdopen(fd).close()
  244. args = [_DEFAULT_GIT, 'daemon', '--verbose', '--export-all',
  245. '--pid-file=%s' % self.pidfile,
  246. '--base-path=%s' % self.gitroot,
  247. '--enable=receive-pack', '--enable=upload-archive',
  248. '--listen=localhost', '--reuseaddr',
  249. self.gitroot]
  250. self.process = subprocess.Popen(
  251. args, env=env, cwd=self.gitroot,
  252. stdout=subprocess.PIPE, stderr=subprocess.PIPE)
  253. if not check_for_daemon():
  254. raise SkipTest('git-daemon failed to start')
  255. def tearDown(self):
  256. with open(self.pidfile) as f:
  257. pid = int(f.read().strip())
  258. if sys.platform == 'win32':
  259. PROCESS_TERMINATE = 1
  260. handle = ctypes.windll.kernel32.OpenProcess(
  261. PROCESS_TERMINATE, False, pid)
  262. ctypes.windll.kernel32.TerminateProcess(handle, -1)
  263. ctypes.windll.kernel32.CloseHandle(handle)
  264. else:
  265. try:
  266. os.kill(pid, signal.SIGKILL)
  267. os.unlink(self.pidfile)
  268. except (OSError, IOError):
  269. pass
  270. self.process.wait()
  271. self.process.stdout.close()
  272. self.process.stderr.close()
  273. DulwichClientTestBase.tearDown(self)
  274. CompatTestCase.tearDown(self)
  275. def _client(self):
  276. return client.TCPGitClient(b'localhost')
  277. def _build_path(self, path):
  278. return path.encode(sys.getfilesystemencoding())
  279. if sys.platform == 'win32':
  280. @expectedFailure
  281. def test_fetch_pack_no_side_band_64k(self):
  282. DulwichClientTestBase.test_fetch_pack_no_side_band_64k(self)
  283. class TestSSHVendor(object):
  284. @staticmethod
  285. def run_command(host, command, username=None, port=None):
  286. cmd, path = command
  287. cmd = cmd.split('-', 1)
  288. p = subprocess.Popen(cmd + [path], bufsize=0, env=get_safe_env(), stdin=subprocess.PIPE,
  289. stdout=subprocess.PIPE, stderr=subprocess.PIPE)
  290. return client.SubprocessWrapper(p)
  291. class DulwichMockSSHClientTest(CompatTestCase, DulwichClientTestBase):
  292. def setUp(self):
  293. CompatTestCase.setUp(self)
  294. DulwichClientTestBase.setUp(self)
  295. self.real_vendor = client.get_ssh_vendor
  296. client.get_ssh_vendor = TestSSHVendor
  297. def tearDown(self):
  298. DulwichClientTestBase.tearDown(self)
  299. CompatTestCase.tearDown(self)
  300. client.get_ssh_vendor = self.real_vendor
  301. def _client(self):
  302. return client.SSHGitClient(b'localhost')
  303. def _build_path(self, path):
  304. return self.gitroot + path
  305. class DulwichSubprocessClientTest(CompatTestCase, DulwichClientTestBase):
  306. def setUp(self):
  307. CompatTestCase.setUp(self)
  308. DulwichClientTestBase.setUp(self)
  309. def tearDown(self):
  310. DulwichClientTestBase.tearDown(self)
  311. CompatTestCase.tearDown(self)
  312. def _client(self):
  313. return client.SubprocessGitClient(stderr=subprocess.PIPE)
  314. def _build_path(self, path):
  315. return self.gitroot + path
  316. class GitHTTPRequestHandler(SimpleHTTPServer.SimpleHTTPRequestHandler):
  317. """HTTP Request handler that calls out to 'git http-backend'."""
  318. # Make rfile unbuffered -- we need to read one line and then pass
  319. # the rest to a subprocess, so we can't use buffered input.
  320. rbufsize = 0
  321. def do_POST(self):
  322. self.run_backend()
  323. def do_GET(self):
  324. self.run_backend()
  325. def send_head(self):
  326. return self.run_backend()
  327. def log_request(self, code='-', size='-'):
  328. # Let's be quiet, the test suite is noisy enough already
  329. pass
  330. def run_backend(self):
  331. """Call out to git http-backend."""
  332. # Based on CGIHTTPServer.CGIHTTPRequestHandler.run_cgi:
  333. # Copyright (c) 2001-2010 Python Software Foundation; All Rights Reserved
  334. # Licensed under the Python Software Foundation License.
  335. rest = self.path
  336. # find an explicit query string, if present.
  337. i = rest.rfind('?')
  338. if i >= 0:
  339. rest, query = rest[:i], rest[i+1:]
  340. else:
  341. query = ''
  342. env = copy.deepcopy(os.environ)
  343. env['SERVER_SOFTWARE'] = self.version_string()
  344. env['SERVER_NAME'] = self.server.server_name
  345. env['GATEWAY_INTERFACE'] = 'CGI/1.1'
  346. env['SERVER_PROTOCOL'] = self.protocol_version
  347. env['SERVER_PORT'] = str(self.server.server_port)
  348. env['GIT_PROJECT_ROOT'] = self.server.root_path
  349. env["GIT_HTTP_EXPORT_ALL"] = "1"
  350. env['REQUEST_METHOD'] = self.command
  351. uqrest = unquote(rest)
  352. env['PATH_INFO'] = uqrest
  353. env['SCRIPT_NAME'] = "/"
  354. if query:
  355. env['QUERY_STRING'] = query
  356. host = self.address_string()
  357. if host != self.client_address[0]:
  358. env['REMOTE_HOST'] = host
  359. env['REMOTE_ADDR'] = self.client_address[0]
  360. authorization = self.headers.get("authorization")
  361. if authorization:
  362. authorization = authorization.split()
  363. if len(authorization) == 2:
  364. import base64, binascii
  365. env['AUTH_TYPE'] = authorization[0]
  366. if authorization[0].lower() == "basic":
  367. try:
  368. authorization = base64.decodestring(authorization[1])
  369. except binascii.Error:
  370. pass
  371. else:
  372. authorization = authorization.split(':')
  373. if len(authorization) == 2:
  374. env['REMOTE_USER'] = authorization[0]
  375. # XXX REMOTE_IDENT
  376. if self.headers.typeheader is None:
  377. env['CONTENT_TYPE'] = self.headers.type
  378. else:
  379. env['CONTENT_TYPE'] = self.headers.typeheader
  380. length = self.headers.get('content-length')
  381. if length:
  382. env['CONTENT_LENGTH'] = length
  383. referer = self.headers.get('referer')
  384. if referer:
  385. env['HTTP_REFERER'] = referer
  386. accept = []
  387. for line in self.headers.getallmatchingheaders('accept'):
  388. if line[:1] in "\t\n\r ":
  389. accept.append(line.strip())
  390. else:
  391. accept = accept + line[7:].split(',')
  392. env['HTTP_ACCEPT'] = ','.join(accept)
  393. ua = self.headers.get('user-agent')
  394. if ua:
  395. env['HTTP_USER_AGENT'] = ua
  396. co = filter(None, self.headers.getheaders('cookie'))
  397. if co:
  398. env['HTTP_COOKIE'] = ', '.join(co)
  399. # XXX Other HTTP_* headers
  400. # Since we're setting the env in the parent, provide empty
  401. # values to override previously set values
  402. for k in ('QUERY_STRING', 'REMOTE_HOST', 'CONTENT_LENGTH',
  403. 'HTTP_USER_AGENT', 'HTTP_COOKIE', 'HTTP_REFERER'):
  404. env.setdefault(k, "")
  405. self.send_response(200, "Script output follows")
  406. decoded_query = query.replace('+', ' ')
  407. try:
  408. nbytes = int(length)
  409. except (TypeError, ValueError):
  410. nbytes = 0
  411. if self.command.lower() == "post" and nbytes > 0:
  412. data = self.rfile.read(nbytes)
  413. else:
  414. data = None
  415. # throw away additional data [see bug #427345]
  416. while select.select([self.rfile._sock], [], [], 0)[0]:
  417. if not self.rfile._sock.recv(1):
  418. break
  419. args = ['http-backend']
  420. if '=' not in decoded_query:
  421. args.append(decoded_query)
  422. stdout = run_git_or_fail(args, input=data, env=env, stderr=subprocess.PIPE)
  423. self.wfile.write(stdout)
  424. class HTTPGitServer(BaseHTTPServer.HTTPServer):
  425. allow_reuse_address = True
  426. def __init__(self, server_address, root_path):
  427. BaseHTTPServer.HTTPServer.__init__(self, server_address, GitHTTPRequestHandler)
  428. self.root_path = root_path
  429. self.server_name = "localhost"
  430. def get_url(self):
  431. return 'http://%s:%s/' % (self.server_name, self.server_port)
  432. @skipIfPY3
  433. class DulwichHttpClientTest(CompatTestCase, DulwichClientTestBase):
  434. min_git_version = (1, 7, 0, 2)
  435. def setUp(self):
  436. CompatTestCase.setUp(self)
  437. DulwichClientTestBase.setUp(self)
  438. self._httpd = HTTPGitServer(("localhost", 0), self.gitroot)
  439. self.addCleanup(self._httpd.shutdown)
  440. threading.Thread(target=self._httpd.serve_forever).start()
  441. run_git_or_fail(['config', 'http.uploadpack', 'true'],
  442. cwd=self.dest)
  443. run_git_or_fail(['config', 'http.receivepack', 'true'],
  444. cwd=self.dest)
  445. def tearDown(self):
  446. DulwichClientTestBase.tearDown(self)
  447. CompatTestCase.tearDown(self)
  448. self._httpd.shutdown()
  449. self._httpd.socket.close()
  450. def _client(self):
  451. return client.HttpGitClient(self._httpd.get_url())
  452. def _build_path(self, path):
  453. return path
  454. def test_archive(self):
  455. raise SkipTest("exporting archives not supported over http")