test_client.py 30 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841
  1. # test_client.py -- Tests for the git protocol, client side
  2. # Copyright (C) 2009 Jelmer Vernooij <jelmer@samba.org>
  3. #
  4. # Dulwich is dual-licensed under the Apache License, Version 2.0 and the GNU
  5. # General Public License as public by the Free Software Foundation; version 2.0
  6. # or (at your option) any later version. You can redistribute it and/or
  7. # modify it under the terms of either of these two licenses.
  8. #
  9. # Unless required by applicable law or agreed to in writing, software
  10. # distributed under the License is distributed on an "AS IS" BASIS,
  11. # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  12. # See the License for the specific language governing permissions and
  13. # limitations under the License.
  14. #
  15. # You should have received a copy of the licenses; if not, see
  16. # <http://www.gnu.org/licenses/> for a copy of the GNU General Public License
  17. # and <http://www.apache.org/licenses/LICENSE-2.0> for a copy of the Apache
  18. # License, Version 2.0.
  19. #
  20. from contextlib import closing
  21. from io import BytesIO
  22. import sys
  23. import shutil
  24. import tempfile
  25. import dulwich
  26. from dulwich import (
  27. client,
  28. )
  29. from dulwich.client import (
  30. LocalGitClient,
  31. TraditionalGitClient,
  32. TCPGitClient,
  33. SSHGitClient,
  34. HttpGitClient,
  35. ReportStatusParser,
  36. SendPackError,
  37. UpdateRefsError,
  38. get_transport_and_path,
  39. get_transport_and_path_from_url,
  40. )
  41. from dulwich.tests import (
  42. TestCase,
  43. )
  44. from dulwich.protocol import (
  45. TCP_GIT_PORT,
  46. Protocol,
  47. )
  48. from dulwich.pack import (
  49. write_pack_objects,
  50. )
  51. from dulwich.objects import (
  52. Commit,
  53. Tree
  54. )
  55. from dulwich.repo import (
  56. MemoryRepo,
  57. Repo,
  58. )
  59. from dulwich.tests import skipIf
  60. from dulwich.tests.utils import (
  61. open_repo,
  62. tear_down_repo,
  63. )
  64. class DummyClient(TraditionalGitClient):
  65. def __init__(self, can_read, read, write):
  66. self.can_read = can_read
  67. self.read = read
  68. self.write = write
  69. TraditionalGitClient.__init__(self)
  70. def _connect(self, service, path):
  71. return Protocol(self.read, self.write), self.can_read
  72. # TODO(durin42): add unit-level tests of GitClient
  73. class GitClientTests(TestCase):
  74. def setUp(self):
  75. super(GitClientTests, self).setUp()
  76. self.rout = BytesIO()
  77. self.rin = BytesIO()
  78. self.client = DummyClient(lambda x: True, self.rin.read,
  79. self.rout.write)
  80. def test_caps(self):
  81. agent_cap = ('agent=dulwich/%d.%d.%d' % dulwich.__version__).encode('ascii')
  82. self.assertEqual(set([b'multi_ack', b'side-band-64k', b'ofs-delta',
  83. b'thin-pack', b'multi_ack_detailed',
  84. agent_cap]),
  85. set(self.client._fetch_capabilities))
  86. self.assertEqual(set([b'ofs-delta', b'report-status', b'side-band-64k',
  87. agent_cap]),
  88. set(self.client._send_capabilities))
  89. def test_archive_ack(self):
  90. self.rin.write(
  91. b'0009NACK\n'
  92. b'0000')
  93. self.rin.seek(0)
  94. self.client.archive(b'bla', b'HEAD', None, None)
  95. self.assertEqual(self.rout.getvalue(), b'0011argument HEAD0000')
  96. def test_fetch_empty(self):
  97. self.rin.write(b'0000')
  98. self.rin.seek(0)
  99. def check_heads(heads):
  100. self.assertIs(heads, None)
  101. return []
  102. ret = self.client.fetch_pack(b'/', check_heads, None, None)
  103. self.assertIs(None, ret)
  104. def test_fetch_pack_ignores_magic_ref(self):
  105. self.rin.write(
  106. b'00000000000000000000000000000000000000000000 capabilities^{}\x00 multi_ack '
  107. b'thin-pack side-band side-band-64k ofs-delta shallow no-progress '
  108. b'include-tag\n'
  109. b'0000')
  110. self.rin.seek(0)
  111. def check_heads(heads):
  112. self.assertEquals({}, heads)
  113. return []
  114. ret = self.client.fetch_pack(b'bla', check_heads, None, None, None)
  115. self.assertIs(None, ret)
  116. self.assertEqual(self.rout.getvalue(), b'0000')
  117. def test_fetch_pack_none(self):
  118. self.rin.write(
  119. b'008855dcc6bf963f922e1ed5c4bbaaefcfacef57b1d7 HEAD.multi_ack '
  120. b'thin-pack side-band side-band-64k ofs-delta shallow no-progress '
  121. b'include-tag\n'
  122. b'0000')
  123. self.rin.seek(0)
  124. self.client.fetch_pack(b'bla', lambda heads: [], None, None, None)
  125. self.assertEqual(self.rout.getvalue(), b'0000')
  126. def test_send_pack_no_sideband64k_with_update_ref_error(self):
  127. # No side-bank-64k reported by server shouldn't try to parse
  128. # side band data
  129. pkts = [b'55dcc6bf963f922e1ed5c4bbaaefcfacef57b1d7 capabilities^{}'
  130. b'\x00 report-status delete-refs ofs-delta\n',
  131. b'',
  132. b"unpack ok",
  133. b"ng refs/foo/bar pre-receive hook declined",
  134. b'']
  135. for pkt in pkts:
  136. if pkt == b'':
  137. self.rin.write(b"0000")
  138. else:
  139. self.rin.write(("%04x" % (len(pkt)+4)).encode('ascii') + pkt)
  140. self.rin.seek(0)
  141. tree = Tree()
  142. commit = Commit()
  143. commit.tree = tree
  144. commit.parents = []
  145. commit.author = commit.committer = b'test user'
  146. commit.commit_time = commit.author_time = 1174773719
  147. commit.commit_timezone = commit.author_timezone = 0
  148. commit.encoding = b'UTF-8'
  149. commit.message = b'test message'
  150. def determine_wants(refs):
  151. return {b'refs/foo/bar': commit.id, }
  152. def generate_pack_contents(have, want):
  153. return [(commit, None), (tree, ''), ]
  154. self.assertRaises(UpdateRefsError,
  155. self.client.send_pack, "blah",
  156. determine_wants, generate_pack_contents)
  157. def test_send_pack_none(self):
  158. self.rin.write(
  159. b'0078310ca9477129b8586fa2afc779c1f57cf64bba6c '
  160. b'refs/heads/master\x00 report-status delete-refs '
  161. b'side-band-64k quiet ofs-delta\n'
  162. b'0000')
  163. self.rin.seek(0)
  164. def determine_wants(refs):
  165. return {
  166. b'refs/heads/master': b'310ca9477129b8586fa2afc779c1f57cf64bba6c'
  167. }
  168. def generate_pack_contents(have, want):
  169. return {}
  170. self.client.send_pack(b'/', determine_wants, generate_pack_contents)
  171. self.assertEqual(self.rout.getvalue(), b'0000')
  172. def test_send_pack_keep_and_delete(self):
  173. self.rin.write(
  174. b'0063310ca9477129b8586fa2afc779c1f57cf64bba6c '
  175. b'refs/heads/master\x00report-status delete-refs ofs-delta\n'
  176. b'003f310ca9477129b8586fa2afc779c1f57cf64bba6c refs/heads/keepme\n'
  177. b'0000000eunpack ok\n'
  178. b'0019ok refs/heads/master\n'
  179. b'0000')
  180. self.rin.seek(0)
  181. def determine_wants(refs):
  182. return {b'refs/heads/master': b'0' * 40}
  183. def generate_pack_contents(have, want):
  184. return {}
  185. self.client.send_pack(b'/', determine_wants, generate_pack_contents)
  186. self.assertIn(
  187. self.rout.getvalue(),
  188. [b'007f310ca9477129b8586fa2afc779c1f57cf64bba6c '
  189. b'0000000000000000000000000000000000000000 '
  190. b'refs/heads/master\x00report-status ofs-delta0000',
  191. b'007f310ca9477129b8586fa2afc779c1f57cf64bba6c '
  192. b'0000000000000000000000000000000000000000 '
  193. b'refs/heads/master\x00ofs-delta report-status0000'])
  194. def test_send_pack_delete_only(self):
  195. self.rin.write(
  196. b'0063310ca9477129b8586fa2afc779c1f57cf64bba6c '
  197. b'refs/heads/master\x00report-status delete-refs ofs-delta\n'
  198. b'0000000eunpack ok\n'
  199. b'0019ok refs/heads/master\n'
  200. b'0000')
  201. self.rin.seek(0)
  202. def determine_wants(refs):
  203. return {b'refs/heads/master': b'0' * 40}
  204. def generate_pack_contents(have, want):
  205. return {}
  206. self.client.send_pack(b'/', determine_wants, generate_pack_contents)
  207. self.assertIn(
  208. self.rout.getvalue(),
  209. [b'007f310ca9477129b8586fa2afc779c1f57cf64bba6c '
  210. b'0000000000000000000000000000000000000000 '
  211. b'refs/heads/master\x00report-status ofs-delta0000',
  212. b'007f310ca9477129b8586fa2afc779c1f57cf64bba6c '
  213. b'0000000000000000000000000000000000000000 '
  214. b'refs/heads/master\x00ofs-delta report-status0000'])
  215. def test_send_pack_new_ref_only(self):
  216. self.rin.write(
  217. b'0063310ca9477129b8586fa2afc779c1f57cf64bba6c '
  218. b'refs/heads/master\x00report-status delete-refs ofs-delta\n'
  219. b'0000000eunpack ok\n'
  220. b'0019ok refs/heads/blah12\n'
  221. b'0000')
  222. self.rin.seek(0)
  223. def determine_wants(refs):
  224. return {
  225. b'refs/heads/blah12':
  226. b'310ca9477129b8586fa2afc779c1f57cf64bba6c',
  227. b'refs/heads/master': b'310ca9477129b8586fa2afc779c1f57cf64bba6c'
  228. }
  229. def generate_pack_contents(have, want):
  230. return {}
  231. f = BytesIO()
  232. write_pack_objects(f, {})
  233. self.client.send_pack('/', determine_wants, generate_pack_contents)
  234. self.assertIn(
  235. self.rout.getvalue(),
  236. [b'007f0000000000000000000000000000000000000000 '
  237. b'310ca9477129b8586fa2afc779c1f57cf64bba6c '
  238. b'refs/heads/blah12\x00report-status ofs-delta0000' +
  239. f.getvalue(),
  240. b'007f0000000000000000000000000000000000000000 '
  241. b'310ca9477129b8586fa2afc779c1f57cf64bba6c '
  242. b'refs/heads/blah12\x00ofs-delta report-status0000' +
  243. f.getvalue()])
  244. def test_send_pack_new_ref(self):
  245. self.rin.write(
  246. b'0064310ca9477129b8586fa2afc779c1f57cf64bba6c '
  247. b'refs/heads/master\x00 report-status delete-refs ofs-delta\n'
  248. b'0000000eunpack ok\n'
  249. b'0019ok refs/heads/blah12\n'
  250. b'0000')
  251. self.rin.seek(0)
  252. tree = Tree()
  253. commit = Commit()
  254. commit.tree = tree
  255. commit.parents = []
  256. commit.author = commit.committer = b'test user'
  257. commit.commit_time = commit.author_time = 1174773719
  258. commit.commit_timezone = commit.author_timezone = 0
  259. commit.encoding = b'UTF-8'
  260. commit.message = b'test message'
  261. def determine_wants(refs):
  262. return {
  263. b'refs/heads/blah12': commit.id,
  264. b'refs/heads/master': b'310ca9477129b8586fa2afc779c1f57cf64bba6c'
  265. }
  266. def generate_pack_contents(have, want):
  267. return [(commit, None), (tree, b''), ]
  268. f = BytesIO()
  269. write_pack_objects(f, generate_pack_contents(None, None))
  270. self.client.send_pack(b'/', determine_wants, generate_pack_contents)
  271. self.assertIn(
  272. self.rout.getvalue(),
  273. [b'007f0000000000000000000000000000000000000000 ' + commit.id +
  274. b' refs/heads/blah12\x00report-status ofs-delta0000' + f.getvalue(),
  275. b'007f0000000000000000000000000000000000000000 ' + commit.id +
  276. b' refs/heads/blah12\x00ofs-delta report-status0000' + f.getvalue()])
  277. def test_send_pack_no_deleteref_delete_only(self):
  278. pkts = [b'310ca9477129b8586fa2afc779c1f57cf64bba6c refs/heads/master'
  279. b'\x00 report-status ofs-delta\n',
  280. b'',
  281. b'']
  282. for pkt in pkts:
  283. if pkt == b'':
  284. self.rin.write(b"0000")
  285. else:
  286. self.rin.write(("%04x" % (len(pkt)+4)).encode('ascii') + pkt)
  287. self.rin.seek(0)
  288. def determine_wants(refs):
  289. return {b'refs/heads/master': b'0' * 40}
  290. def generate_pack_contents(have, want):
  291. return {}
  292. self.assertRaises(UpdateRefsError,
  293. self.client.send_pack, b"/",
  294. determine_wants, generate_pack_contents)
  295. self.assertEqual(self.rout.getvalue(), b'0000')
  296. class TestGetTransportAndPath(TestCase):
  297. def test_tcp(self):
  298. c, path = get_transport_and_path('git://foo.com/bar/baz')
  299. self.assertTrue(isinstance(c, TCPGitClient))
  300. self.assertEqual('foo.com', c._host)
  301. self.assertEqual(TCP_GIT_PORT, c._port)
  302. self.assertEqual('/bar/baz', path)
  303. def test_tcp_port(self):
  304. c, path = get_transport_and_path('git://foo.com:1234/bar/baz')
  305. self.assertTrue(isinstance(c, TCPGitClient))
  306. self.assertEqual('foo.com', c._host)
  307. self.assertEqual(1234, c._port)
  308. self.assertEqual('/bar/baz', path)
  309. def test_git_ssh_explicit(self):
  310. c, path = get_transport_and_path('git+ssh://foo.com/bar/baz')
  311. self.assertTrue(isinstance(c, SSHGitClient))
  312. self.assertEqual('foo.com', c.host)
  313. self.assertEqual(None, c.port)
  314. self.assertEqual(None, c.username)
  315. self.assertEqual('bar/baz', path)
  316. def test_ssh_explicit(self):
  317. c, path = get_transport_and_path('ssh://foo.com/bar/baz')
  318. self.assertTrue(isinstance(c, SSHGitClient))
  319. self.assertEqual('foo.com', c.host)
  320. self.assertEqual(None, c.port)
  321. self.assertEqual(None, c.username)
  322. self.assertEqual('bar/baz', path)
  323. def test_ssh_port_explicit(self):
  324. c, path = get_transport_and_path(
  325. 'git+ssh://foo.com:1234/bar/baz')
  326. self.assertTrue(isinstance(c, SSHGitClient))
  327. self.assertEqual('foo.com', c.host)
  328. self.assertEqual(1234, c.port)
  329. self.assertEqual('bar/baz', path)
  330. def test_username_and_port_explicit_unknown_scheme(self):
  331. c, path = get_transport_and_path(
  332. 'unknown://git@server:7999/dply/stuff.git')
  333. self.assertTrue(isinstance(c, SSHGitClient))
  334. self.assertEqual('unknown', c.host)
  335. self.assertEqual('//git@server:7999/dply/stuff.git', path)
  336. def test_username_and_port_explicit(self):
  337. c, path = get_transport_and_path(
  338. 'ssh://git@server:7999/dply/stuff.git')
  339. self.assertTrue(isinstance(c, SSHGitClient))
  340. self.assertEqual('git', c.username)
  341. self.assertEqual('server', c.host)
  342. self.assertEqual(7999, c.port)
  343. self.assertEqual('dply/stuff.git', path)
  344. def test_ssh_abspath_explicit(self):
  345. c, path = get_transport_and_path('git+ssh://foo.com//bar/baz')
  346. self.assertTrue(isinstance(c, SSHGitClient))
  347. self.assertEqual('foo.com', c.host)
  348. self.assertEqual(None, c.port)
  349. self.assertEqual(None, c.username)
  350. self.assertEqual('/bar/baz', path)
  351. def test_ssh_port_abspath_explicit(self):
  352. c, path = get_transport_and_path(
  353. 'git+ssh://foo.com:1234//bar/baz')
  354. self.assertTrue(isinstance(c, SSHGitClient))
  355. self.assertEqual('foo.com', c.host)
  356. self.assertEqual(1234, c.port)
  357. self.assertEqual('/bar/baz', path)
  358. def test_ssh_implicit(self):
  359. c, path = get_transport_and_path('foo:/bar/baz')
  360. self.assertTrue(isinstance(c, SSHGitClient))
  361. self.assertEqual('foo', c.host)
  362. self.assertEqual(None, c.port)
  363. self.assertEqual(None, c.username)
  364. self.assertEqual('/bar/baz', path)
  365. def test_ssh_host(self):
  366. c, path = get_transport_and_path('foo.com:/bar/baz')
  367. self.assertTrue(isinstance(c, SSHGitClient))
  368. self.assertEqual('foo.com', c.host)
  369. self.assertEqual(None, c.port)
  370. self.assertEqual(None, c.username)
  371. self.assertEqual('/bar/baz', path)
  372. def test_ssh_user_host(self):
  373. c, path = get_transport_and_path('user@foo.com:/bar/baz')
  374. self.assertTrue(isinstance(c, SSHGitClient))
  375. self.assertEqual('foo.com', c.host)
  376. self.assertEqual(None, c.port)
  377. self.assertEqual('user', c.username)
  378. self.assertEqual('/bar/baz', path)
  379. def test_ssh_relpath(self):
  380. c, path = get_transport_and_path('foo:bar/baz')
  381. self.assertTrue(isinstance(c, SSHGitClient))
  382. self.assertEqual('foo', c.host)
  383. self.assertEqual(None, c.port)
  384. self.assertEqual(None, c.username)
  385. self.assertEqual('bar/baz', path)
  386. def test_ssh_host_relpath(self):
  387. c, path = get_transport_and_path('foo.com:bar/baz')
  388. self.assertTrue(isinstance(c, SSHGitClient))
  389. self.assertEqual('foo.com', c.host)
  390. self.assertEqual(None, c.port)
  391. self.assertEqual(None, c.username)
  392. self.assertEqual('bar/baz', path)
  393. def test_ssh_user_host_relpath(self):
  394. c, path = get_transport_and_path('user@foo.com:bar/baz')
  395. self.assertTrue(isinstance(c, SSHGitClient))
  396. self.assertEqual('foo.com', c.host)
  397. self.assertEqual(None, c.port)
  398. self.assertEqual('user', c.username)
  399. self.assertEqual('bar/baz', path)
  400. def test_local(self):
  401. c, path = get_transport_and_path('foo.bar/baz')
  402. self.assertTrue(isinstance(c, LocalGitClient))
  403. self.assertEqual('foo.bar/baz', path)
  404. @skipIf(sys.platform != 'win32', 'Behaviour only happens on windows.')
  405. def test_local_abs_windows_path(self):
  406. c, path = get_transport_and_path('C:\\foo.bar\\baz')
  407. self.assertTrue(isinstance(c, LocalGitClient))
  408. self.assertEqual('C:\\foo.bar\\baz', path)
  409. def test_error(self):
  410. # Need to use a known urlparse.uses_netloc URL scheme to get the
  411. # expected parsing of the URL on Python versions less than 2.6.5
  412. c, path = get_transport_and_path('prospero://bar/baz')
  413. self.assertTrue(isinstance(c, SSHGitClient))
  414. def test_http(self):
  415. url = 'https://github.com/jelmer/dulwich'
  416. c, path = get_transport_and_path(url)
  417. self.assertTrue(isinstance(c, HttpGitClient))
  418. self.assertEqual('/jelmer/dulwich', path)
  419. def test_http_auth(self):
  420. url = 'https://user:passwd@github.com/jelmer/dulwich'
  421. c, path = get_transport_and_path(url)
  422. self.assertTrue(isinstance(c, HttpGitClient))
  423. self.assertEqual('/jelmer/dulwich', path)
  424. self.assertEqual('user', c._username)
  425. self.assertEqual('passwd', c._password)
  426. def test_http_no_auth(self):
  427. url = 'https://github.com/jelmer/dulwich'
  428. c, path = get_transport_and_path(url)
  429. self.assertTrue(isinstance(c, HttpGitClient))
  430. self.assertEqual('/jelmer/dulwich', path)
  431. self.assertIs(None, c._username)
  432. self.assertIs(None, c._password)
  433. class TestGetTransportAndPathFromUrl(TestCase):
  434. def test_tcp(self):
  435. c, path = get_transport_and_path_from_url('git://foo.com/bar/baz')
  436. self.assertTrue(isinstance(c, TCPGitClient))
  437. self.assertEqual('foo.com', c._host)
  438. self.assertEqual(TCP_GIT_PORT, c._port)
  439. self.assertEqual('/bar/baz', path)
  440. def test_tcp_port(self):
  441. c, path = get_transport_and_path_from_url('git://foo.com:1234/bar/baz')
  442. self.assertTrue(isinstance(c, TCPGitClient))
  443. self.assertEqual('foo.com', c._host)
  444. self.assertEqual(1234, c._port)
  445. self.assertEqual('/bar/baz', path)
  446. def test_ssh_explicit(self):
  447. c, path = get_transport_and_path_from_url('git+ssh://foo.com/bar/baz')
  448. self.assertTrue(isinstance(c, SSHGitClient))
  449. self.assertEqual('foo.com', c.host)
  450. self.assertEqual(None, c.port)
  451. self.assertEqual(None, c.username)
  452. self.assertEqual('bar/baz', path)
  453. def test_ssh_port_explicit(self):
  454. c, path = get_transport_and_path_from_url(
  455. 'git+ssh://foo.com:1234/bar/baz')
  456. self.assertTrue(isinstance(c, SSHGitClient))
  457. self.assertEqual('foo.com', c.host)
  458. self.assertEqual(1234, c.port)
  459. self.assertEqual('bar/baz', path)
  460. def test_ssh_abspath_explicit(self):
  461. c, path = get_transport_and_path_from_url('git+ssh://foo.com//bar/baz')
  462. self.assertTrue(isinstance(c, SSHGitClient))
  463. self.assertEqual('foo.com', c.host)
  464. self.assertEqual(None, c.port)
  465. self.assertEqual(None, c.username)
  466. self.assertEqual('/bar/baz', path)
  467. def test_ssh_port_abspath_explicit(self):
  468. c, path = get_transport_and_path_from_url(
  469. 'git+ssh://foo.com:1234//bar/baz')
  470. self.assertTrue(isinstance(c, SSHGitClient))
  471. self.assertEqual('foo.com', c.host)
  472. self.assertEqual(1234, c.port)
  473. self.assertEqual('/bar/baz', path)
  474. def test_ssh_host_relpath(self):
  475. self.assertRaises(ValueError, get_transport_and_path_from_url,
  476. 'foo.com:bar/baz')
  477. def test_ssh_user_host_relpath(self):
  478. self.assertRaises(ValueError, get_transport_and_path_from_url,
  479. 'user@foo.com:bar/baz')
  480. def test_local_path(self):
  481. self.assertRaises(ValueError, get_transport_and_path_from_url,
  482. 'foo.bar/baz')
  483. def test_error(self):
  484. # Need to use a known urlparse.uses_netloc URL scheme to get the
  485. # expected parsing of the URL on Python versions less than 2.6.5
  486. self.assertRaises(ValueError, get_transport_and_path_from_url,
  487. 'prospero://bar/baz')
  488. def test_http(self):
  489. url = 'https://github.com/jelmer/dulwich'
  490. c, path = get_transport_and_path_from_url(url)
  491. self.assertTrue(isinstance(c, HttpGitClient))
  492. self.assertEqual('/jelmer/dulwich', path)
  493. def test_file(self):
  494. c, path = get_transport_and_path_from_url('file:///home/jelmer/foo')
  495. self.assertTrue(isinstance(c, LocalGitClient))
  496. self.assertEqual('/home/jelmer/foo', path)
  497. class TestSSHVendor(object):
  498. def __init__(self):
  499. self.host = None
  500. self.command = ""
  501. self.username = None
  502. self.port = None
  503. def run_command(self, host, command, username=None, port=None):
  504. if not isinstance(command, bytes):
  505. raise TypeError(command)
  506. self.host = host
  507. self.command = command
  508. self.username = username
  509. self.port = port
  510. class Subprocess: pass
  511. setattr(Subprocess, 'read', lambda: None)
  512. setattr(Subprocess, 'write', lambda: None)
  513. setattr(Subprocess, 'close', lambda: None)
  514. setattr(Subprocess, 'can_read', lambda: None)
  515. return Subprocess()
  516. class SSHGitClientTests(TestCase):
  517. def setUp(self):
  518. super(SSHGitClientTests, self).setUp()
  519. self.server = TestSSHVendor()
  520. self.real_vendor = client.get_ssh_vendor
  521. client.get_ssh_vendor = lambda: self.server
  522. self.client = SSHGitClient('git.samba.org')
  523. def tearDown(self):
  524. super(SSHGitClientTests, self).tearDown()
  525. client.get_ssh_vendor = self.real_vendor
  526. def test_get_url(self):
  527. path = '/tmp/repo.git'
  528. c = SSHGitClient('git.samba.org')
  529. url = c.get_url(path)
  530. self.assertEqual('ssh://git.samba.org/tmp/repo.git', url)
  531. def test_get_url_with_username_and_port(self):
  532. path = '/tmp/repo.git'
  533. c = SSHGitClient('git.samba.org', port=2222, username='user')
  534. url = c.get_url(path)
  535. self.assertEqual('ssh://user@git.samba.org:2222/tmp/repo.git', url)
  536. def test_default_command(self):
  537. self.assertEqual(b'git-upload-pack',
  538. self.client._get_cmd_path(b'upload-pack'))
  539. def test_alternative_command_path(self):
  540. self.client.alternative_paths[b'upload-pack'] = (
  541. b'/usr/lib/git/git-upload-pack')
  542. self.assertEqual(b'/usr/lib/git/git-upload-pack',
  543. self.client._get_cmd_path(b'upload-pack'))
  544. def test_alternative_command_path_spaces(self):
  545. self.client.alternative_paths[b'upload-pack'] = (
  546. b'/usr/lib/git/git-upload-pack -ibla')
  547. self.assertEqual(b"/usr/lib/git/git-upload-pack -ibla",
  548. self.client._get_cmd_path(b'upload-pack'))
  549. def test_connect(self):
  550. server = self.server
  551. client = self.client
  552. client.username = b"username"
  553. client.port = 1337
  554. client._connect(b"command", b"/path/to/repo")
  555. self.assertEqual(b"username", server.username)
  556. self.assertEqual(1337, server.port)
  557. self.assertEqual(b"git-command '/path/to/repo'", server.command)
  558. client._connect(b"relative-command", b"/~/path/to/repo")
  559. self.assertEqual(b"git-relative-command '~/path/to/repo'",
  560. server.command)
  561. class ReportStatusParserTests(TestCase):
  562. def test_invalid_pack(self):
  563. parser = ReportStatusParser()
  564. parser.handle_packet(b"unpack error - foo bar")
  565. parser.handle_packet(b"ok refs/foo/bar")
  566. parser.handle_packet(None)
  567. self.assertRaises(SendPackError, parser.check)
  568. def test_update_refs_error(self):
  569. parser = ReportStatusParser()
  570. parser.handle_packet(b"unpack ok")
  571. parser.handle_packet(b"ng refs/foo/bar need to pull")
  572. parser.handle_packet(None)
  573. self.assertRaises(UpdateRefsError, parser.check)
  574. def test_ok(self):
  575. parser = ReportStatusParser()
  576. parser.handle_packet(b"unpack ok")
  577. parser.handle_packet(b"ok refs/foo/bar")
  578. parser.handle_packet(None)
  579. parser.check()
  580. class LocalGitClientTests(TestCase):
  581. def test_get_url(self):
  582. path = "/tmp/repo.git"
  583. c = LocalGitClient()
  584. url = c.get_url(path)
  585. self.assertEqual('file:///tmp/repo.git', url)
  586. def test_fetch_into_empty(self):
  587. c = LocalGitClient()
  588. t = MemoryRepo()
  589. s = open_repo('a.git')
  590. self.addCleanup(tear_down_repo, s)
  591. self.assertEqual(s.get_refs(), c.fetch(s.path, t))
  592. def test_fetch_empty(self):
  593. c = LocalGitClient()
  594. s = open_repo('a.git')
  595. self.addCleanup(tear_down_repo, s)
  596. out = BytesIO()
  597. walker = {}
  598. ret = c.fetch_pack(s.path, lambda heads: [], graph_walker=walker,
  599. pack_data=out.write)
  600. self.assertEqual({
  601. b'HEAD': b'a90fa2d900a17e99b433217e988c4eb4a2e9a097',
  602. b'refs/heads/master': b'a90fa2d900a17e99b433217e988c4eb4a2e9a097',
  603. b'refs/tags/mytag': b'28237f4dc30d0d462658d6b937b08a0f0b6ef55a',
  604. b'refs/tags/mytag-packed': b'b0931cadc54336e78a1d980420e3268903b57a50'
  605. }, ret)
  606. self.assertEqual(b"PACK\x00\x00\x00\x02\x00\x00\x00\x00\x02\x9d\x08"
  607. b"\x82;\xd8\xa8\xea\xb5\x10\xadj\xc7\\\x82<\xfd>\xd3\x1e", out.getvalue())
  608. def test_fetch_pack_none(self):
  609. c = LocalGitClient()
  610. s = open_repo('a.git')
  611. self.addCleanup(tear_down_repo, s)
  612. out = BytesIO()
  613. walker = MemoryRepo().get_graph_walker()
  614. c.fetch_pack(s.path,
  615. lambda heads: [b"a90fa2d900a17e99b433217e988c4eb4a2e9a097"],
  616. graph_walker=walker, pack_data=out.write)
  617. # Hardcoding is not ideal, but we'll fix that some other day..
  618. self.assertTrue(out.getvalue().startswith(b'PACK\x00\x00\x00\x02\x00\x00\x00\x07'))
  619. def test_send_pack_without_changes(self):
  620. local = open_repo('a.git')
  621. self.addCleanup(tear_down_repo, local)
  622. target = open_repo('a.git')
  623. self.addCleanup(tear_down_repo, target)
  624. self.send_and_verify(b"master", local, target)
  625. def test_send_pack_with_changes(self):
  626. local = open_repo('a.git')
  627. self.addCleanup(tear_down_repo, local)
  628. target_path = tempfile.mkdtemp()
  629. self.addCleanup(shutil.rmtree, target_path)
  630. with closing(Repo.init_bare(target_path)) as target:
  631. self.send_and_verify(b"master", local, target)
  632. def test_get_refs(self):
  633. local = open_repo('refs.git')
  634. self.addCleanup(tear_down_repo, local)
  635. client = LocalGitClient()
  636. refs = client.get_refs(local.path)
  637. self.assertDictEqual(local.refs.as_dict(), refs)
  638. def send_and_verify(self, branch, local, target):
  639. """Send a branch from local to remote repository and verify it worked."""
  640. client = LocalGitClient()
  641. ref_name = b"refs/heads/" + branch
  642. new_refs = client.send_pack(target.path,
  643. lambda _: { ref_name: local.refs[ref_name] },
  644. local.object_store.generate_pack_contents)
  645. self.assertEqual(local.refs[ref_name], new_refs[ref_name])
  646. obj_local = local.get_object(new_refs[ref_name])
  647. obj_target = target.get_object(new_refs[ref_name])
  648. self.assertEqual(obj_local, obj_target)
  649. class HttpGitClientTests(TestCase):
  650. def test_get_url(self):
  651. base_url = 'https://github.com/jelmer/dulwich'
  652. path = '/jelmer/dulwich'
  653. c = HttpGitClient(base_url)
  654. url = c.get_url(path)
  655. self.assertEqual('https://github.com/jelmer/dulwich', url)
  656. def test_get_url_with_username_and_passwd(self):
  657. base_url = 'https://github.com/jelmer/dulwich'
  658. path = '/jelmer/dulwich'
  659. c = HttpGitClient(base_url, username='USERNAME', password='PASSWD')
  660. url = c.get_url(path)
  661. self.assertEqual('https://github.com/jelmer/dulwich', url)
  662. def test_init_username_passwd_set(self):
  663. url = 'https://github.com/jelmer/dulwich'
  664. c = HttpGitClient(url, config=None, username='user', password='passwd')
  665. self.assertEqual('user', c._username)
  666. self.assertEqual('passwd', c._password)
  667. [pw_handler] = [
  668. h for h in c.opener.handlers if getattr(h, 'passwd', None) is not None]
  669. self.assertEqual(
  670. ('user', 'passwd'),
  671. pw_handler.passwd.find_user_password(
  672. None, 'https://github.com/jelmer/dulwich'))
  673. def test_init_no_username_passwd(self):
  674. url = 'https://github.com/jelmer/dulwich'
  675. c = HttpGitClient(url, config=None)
  676. self.assertIs(None, c._username)
  677. self.assertIs(None, c._password)
  678. pw_handler = [
  679. h for h in c.opener.handlers if getattr(h, 'passwd', None) is not None]
  680. self.assertEqual(0, len(pw_handler))
  681. class TCPGitClientTests(TestCase):
  682. def test_get_url(self):
  683. host = 'github.com'
  684. path = '/jelmer/dulwich'
  685. c = TCPGitClient(host)
  686. url = c.get_url(path)
  687. self.assertEqual('git://github.com/jelmer/dulwich', url)
  688. def test_get_url_with_port(self):
  689. host = 'github.com'
  690. path = '/jelmer/dulwich'
  691. port = 9090
  692. c = TCPGitClient(host, port=9090)
  693. url = c.get_url(path)
  694. self.assertEqual('git://github.com:9090/jelmer/dulwich', url)