test_client.py 33 KB

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