test_client.py 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701
  1. # test_client.py -- Compatibility tests for git client.
  2. # Copyright (C) 2010 Google, Inc.
  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. """Compatibility tests between the Dulwich client and the cgit server."""
  21. import copy
  22. import http.server
  23. import os
  24. import select
  25. import signal
  26. import stat
  27. import subprocess
  28. import sys
  29. import tarfile
  30. import tempfile
  31. import threading
  32. from contextlib import suppress
  33. from io import BytesIO
  34. from unittest.mock import patch
  35. from urllib.parse import unquote
  36. from dulwich import client, file, index, objects, protocol, repo
  37. from .. import SkipTest, expectedFailure
  38. from .utils import (
  39. _DEFAULT_GIT,
  40. CompatTestCase,
  41. check_for_daemon,
  42. import_repo_to_dir,
  43. rmtree_ro,
  44. run_git_or_fail,
  45. )
  46. if sys.platform == "win32":
  47. import ctypes
  48. class DulwichClientTestBase:
  49. """Tests for client/server compatibility."""
  50. def setUp(self):
  51. self.gitroot = os.path.dirname(
  52. import_repo_to_dir("server_new.export").rstrip(os.sep)
  53. )
  54. self.dest = os.path.join(self.gitroot, "dest")
  55. file.ensure_dir_exists(self.dest)
  56. run_git_or_fail(["init", "--quiet", "--bare"], cwd=self.dest)
  57. def tearDown(self):
  58. rmtree_ro(self.gitroot)
  59. def assertDestEqualsSrc(self):
  60. repo_dir = os.path.join(self.gitroot, "server_new.export")
  61. dest_repo_dir = os.path.join(self.gitroot, "dest")
  62. with repo.Repo(repo_dir) as src:
  63. with repo.Repo(dest_repo_dir) as dest:
  64. self.assertReposEqual(src, dest)
  65. def _client(self):
  66. raise NotImplementedError
  67. def _build_path(self):
  68. raise NotImplementedError
  69. def _do_send_pack(self):
  70. c = self._client()
  71. srcpath = os.path.join(self.gitroot, "server_new.export")
  72. with repo.Repo(srcpath) as src:
  73. sendrefs = dict(src.get_refs())
  74. del sendrefs[b"HEAD"]
  75. c.send_pack(
  76. self._build_path("/dest"),
  77. lambda _: sendrefs,
  78. src.generate_pack_data,
  79. )
  80. def test_send_pack(self):
  81. self._do_send_pack()
  82. self.assertDestEqualsSrc()
  83. def test_send_pack_nothing_to_send(self):
  84. self._do_send_pack()
  85. self.assertDestEqualsSrc()
  86. # nothing to send, but shouldn't raise either.
  87. self._do_send_pack()
  88. @staticmethod
  89. def _add_file(repo, tree_id, filename, contents):
  90. tree = repo[tree_id]
  91. blob = objects.Blob()
  92. blob.data = contents.encode("utf-8")
  93. repo.object_store.add_object(blob)
  94. tree.add(filename.encode("utf-8"), stat.S_IFREG | 0o644, blob.id)
  95. repo.object_store.add_object(tree)
  96. return tree.id
  97. def test_send_pack_from_shallow_clone(self):
  98. c = self._client()
  99. server_new_path = os.path.join(self.gitroot, "server_new.export")
  100. run_git_or_fail(["config", "http.uploadpack", "true"], cwd=server_new_path)
  101. run_git_or_fail(["config", "http.receivepack", "true"], cwd=server_new_path)
  102. remote_path = self._build_path("/server_new.export")
  103. with repo.Repo(self.dest) as local:
  104. result = c.fetch(remote_path, local, depth=1)
  105. for r in result.refs.items():
  106. local.refs.set_if_equals(r[0], None, r[1])
  107. tree_id = local[local.head()].tree
  108. for filename, contents in [
  109. ("bar", "bar contents"),
  110. ("zop", "zop contents"),
  111. ]:
  112. tree_id = self._add_file(local, tree_id, filename, contents)
  113. commit_id = local.do_commit(
  114. message=b"add " + filename.encode("utf-8"),
  115. committer=b"Joe Example <joe@example.com>",
  116. tree=tree_id,
  117. )
  118. sendrefs = dict(local.get_refs())
  119. del sendrefs[b"HEAD"]
  120. c.send_pack(remote_path, lambda _: sendrefs, local.generate_pack_data)
  121. with repo.Repo(server_new_path) as remote:
  122. self.assertEqual(remote.head(), commit_id)
  123. def test_send_without_report_status(self):
  124. c = self._client()
  125. c._send_capabilities.remove(b"report-status")
  126. srcpath = os.path.join(self.gitroot, "server_new.export")
  127. with repo.Repo(srcpath) as src:
  128. sendrefs = dict(src.get_refs())
  129. del sendrefs[b"HEAD"]
  130. c.send_pack(
  131. self._build_path("/dest"),
  132. lambda _: sendrefs,
  133. src.generate_pack_data,
  134. )
  135. self.assertDestEqualsSrc()
  136. def make_dummy_commit(self, dest):
  137. b = objects.Blob.from_string(b"hi")
  138. dest.object_store.add_object(b)
  139. t = index.commit_tree(dest.object_store, [(b"hi", b.id, 0o100644)])
  140. c = objects.Commit()
  141. c.author = c.committer = b"Foo Bar <foo@example.com>"
  142. c.author_time = c.commit_time = 0
  143. c.author_timezone = c.commit_timezone = 0
  144. c.message = b"hi"
  145. c.tree = t
  146. dest.object_store.add_object(c)
  147. return c.id
  148. def disable_ff_and_make_dummy_commit(self):
  149. # disable non-fast-forward pushes to the server
  150. dest = repo.Repo(os.path.join(self.gitroot, "dest"))
  151. run_git_or_fail(
  152. ["config", "receive.denyNonFastForwards", "true"], cwd=dest.path
  153. )
  154. commit_id = self.make_dummy_commit(dest)
  155. return dest, commit_id
  156. def compute_send(self, src):
  157. sendrefs = dict(src.get_refs())
  158. del sendrefs[b"HEAD"]
  159. return sendrefs, src.generate_pack_data
  160. def test_send_pack_one_error(self):
  161. dest, dummy_commit = self.disable_ff_and_make_dummy_commit()
  162. dest.refs[b"refs/heads/master"] = dummy_commit
  163. repo_dir = os.path.join(self.gitroot, "server_new.export")
  164. with repo.Repo(repo_dir) as src:
  165. sendrefs, gen_pack = self.compute_send(src)
  166. c = self._client()
  167. result = c.send_pack(
  168. self._build_path("/dest"), lambda _: sendrefs, gen_pack
  169. )
  170. self.assertEqual(
  171. {
  172. b"refs/heads/branch": None,
  173. b"refs/heads/master": "non-fast-forward",
  174. },
  175. result.ref_status,
  176. )
  177. def test_send_pack_multiple_errors(self):
  178. dest, dummy = self.disable_ff_and_make_dummy_commit()
  179. # set up for two non-ff errors
  180. branch, master = b"refs/heads/branch", b"refs/heads/master"
  181. dest.refs[branch] = dest.refs[master] = dummy
  182. repo_dir = os.path.join(self.gitroot, "server_new.export")
  183. with repo.Repo(repo_dir) as src:
  184. sendrefs, gen_pack = self.compute_send(src)
  185. c = self._client()
  186. result = c.send_pack(
  187. self._build_path("/dest"), lambda _: sendrefs, gen_pack
  188. )
  189. self.assertEqual(
  190. {branch: "non-fast-forward", master: "non-fast-forward"},
  191. result.ref_status,
  192. )
  193. def test_archive(self):
  194. c = self._client()
  195. f = BytesIO()
  196. c.archive(self._build_path("/server_new.export"), b"HEAD", f.write)
  197. f.seek(0)
  198. tf = tarfile.open(fileobj=f)
  199. self.assertEqual(["baz", "foo"], tf.getnames())
  200. def test_fetch_pack(self):
  201. c = self._client()
  202. with repo.Repo(os.path.join(self.gitroot, "dest")) as dest:
  203. result = c.fetch(self._build_path("/server_new.export"), dest)
  204. self.assertEqual(
  205. {b"HEAD": b"refs/heads/master"},
  206. result.symrefs,
  207. )
  208. for r in result.refs.items():
  209. dest.refs.set_if_equals(r[0], None, r[1])
  210. self.assertDestEqualsSrc()
  211. def test_fetch_pack_depth(self):
  212. c = self._client()
  213. with repo.Repo(os.path.join(self.gitroot, "dest")) as dest:
  214. result = c.fetch(self._build_path("/server_new.export"), dest, depth=1)
  215. for r in result.refs.items():
  216. dest.refs.set_if_equals(r[0], None, r[1])
  217. self.assertEqual(
  218. dest.get_shallow(),
  219. {
  220. b"35e0b59e187dd72a0af294aedffc213eaa4d03ff",
  221. b"514dc6d3fbfe77361bcaef320c4d21b72bc10be9",
  222. },
  223. )
  224. def test_repeat(self):
  225. c = self._client()
  226. with repo.Repo(os.path.join(self.gitroot, "dest")) as dest:
  227. result = c.fetch(self._build_path("/server_new.export"), dest)
  228. for r in result.refs.items():
  229. dest.refs.set_if_equals(r[0], None, r[1])
  230. self.assertDestEqualsSrc()
  231. result = c.fetch(self._build_path("/server_new.export"), dest)
  232. for r in result.refs.items():
  233. dest.refs.set_if_equals(r[0], None, r[1])
  234. self.assertDestEqualsSrc()
  235. def test_fetch_empty_pack(self):
  236. c = self._client()
  237. with repo.Repo(os.path.join(self.gitroot, "dest")) as dest:
  238. result = c.fetch(self._build_path("/server_new.export"), dest)
  239. for r in result.refs.items():
  240. dest.refs.set_if_equals(r[0], None, r[1])
  241. self.assertDestEqualsSrc()
  242. def dw(refs, **kwargs):
  243. return list(refs.values())
  244. result = c.fetch(
  245. self._build_path("/server_new.export"),
  246. dest,
  247. determine_wants=dw,
  248. )
  249. for r in result.refs.items():
  250. dest.refs.set_if_equals(r[0], None, r[1])
  251. self.assertDestEqualsSrc()
  252. def test_incremental_fetch_pack(self):
  253. self.test_fetch_pack()
  254. dest, dummy = self.disable_ff_and_make_dummy_commit()
  255. dest.refs[b"refs/heads/master"] = dummy
  256. c = self._client()
  257. repo_dir = os.path.join(self.gitroot, "server_new.export")
  258. with repo.Repo(repo_dir) as dest:
  259. result = c.fetch(self._build_path("/dest"), dest)
  260. for r in result.refs.items():
  261. dest.refs.set_if_equals(r[0], None, r[1])
  262. self.assertDestEqualsSrc()
  263. def test_fetch_pack_no_side_band_64k(self):
  264. if protocol.DEFAULT_GIT_PROTOCOL_VERSION_FETCH >= 2:
  265. raise SkipTest("side-band-64k cannot be disabled with git protocol v2")
  266. c = self._client()
  267. c._fetch_capabilities.remove(b"side-band-64k")
  268. with repo.Repo(os.path.join(self.gitroot, "dest")) as dest:
  269. result = c.fetch(self._build_path("/server_new.export"), dest)
  270. for r in result.refs.items():
  271. dest.refs.set_if_equals(r[0], None, r[1])
  272. self.assertDestEqualsSrc()
  273. def test_fetch_pack_zero_sha(self):
  274. # zero sha1s are already present on the client, and should
  275. # be ignored
  276. c = self._client()
  277. with repo.Repo(os.path.join(self.gitroot, "dest")) as dest:
  278. result = c.fetch(
  279. self._build_path("/server_new.export"),
  280. dest,
  281. lambda refs, **kwargs: [protocol.ZERO_SHA],
  282. )
  283. for r in result.refs.items():
  284. dest.refs.set_if_equals(r[0], None, r[1])
  285. def test_send_remove_branch(self):
  286. with repo.Repo(os.path.join(self.gitroot, "dest")) as dest:
  287. dummy_commit = self.make_dummy_commit(dest)
  288. dest.refs[b"refs/heads/master"] = dummy_commit
  289. dest.refs[b"refs/heads/abranch"] = dummy_commit
  290. sendrefs = dict(dest.refs)
  291. sendrefs[b"refs/heads/abranch"] = b"00" * 20
  292. del sendrefs[b"HEAD"]
  293. def gen_pack(have, want, ofs_delta=False, progress=None):
  294. return 0, []
  295. c = self._client()
  296. self.assertEqual(dest.refs[b"refs/heads/abranch"], dummy_commit)
  297. c.send_pack(self._build_path("/dest"), lambda _: sendrefs, gen_pack)
  298. self.assertNotIn(b"refs/heads/abranch", dest.refs)
  299. def test_send_new_branch_empty_pack(self):
  300. with repo.Repo(os.path.join(self.gitroot, "dest")) as dest:
  301. dummy_commit = self.make_dummy_commit(dest)
  302. dest.refs[b"refs/heads/master"] = dummy_commit
  303. dest.refs[b"refs/heads/abranch"] = dummy_commit
  304. sendrefs = {b"refs/heads/bbranch": dummy_commit}
  305. def gen_pack(have, want, ofs_delta=False, progress=None):
  306. return 0, []
  307. c = self._client()
  308. self.assertEqual(dest.refs[b"refs/heads/abranch"], dummy_commit)
  309. c.send_pack(self._build_path("/dest"), lambda _: sendrefs, gen_pack)
  310. self.assertEqual(dummy_commit, dest.refs[b"refs/heads/abranch"])
  311. def test_get_refs(self):
  312. c = self._client()
  313. refs = c.get_refs(self._build_path("/server_new.export"))
  314. repo_dir = os.path.join(self.gitroot, "server_new.export")
  315. with repo.Repo(repo_dir) as dest:
  316. self.assertDictEqual(dest.refs.as_dict(), refs)
  317. class DulwichTCPClientTest(CompatTestCase, DulwichClientTestBase):
  318. def setUp(self):
  319. CompatTestCase.setUp(self)
  320. DulwichClientTestBase.setUp(self)
  321. if check_for_daemon(limit=1):
  322. raise SkipTest(
  323. f"git-daemon was already running on port {protocol.TCP_GIT_PORT}"
  324. )
  325. fd, self.pidfile = tempfile.mkstemp(
  326. prefix="dulwich-test-git-client", suffix=".pid"
  327. )
  328. os.fdopen(fd).close()
  329. args = [
  330. _DEFAULT_GIT,
  331. "daemon",
  332. "--verbose",
  333. "--export-all",
  334. f"--pid-file={self.pidfile}",
  335. f"--base-path={self.gitroot}",
  336. "--enable=receive-pack",
  337. "--enable=upload-archive",
  338. "--listen=localhost",
  339. "--reuseaddr",
  340. self.gitroot,
  341. ]
  342. self.process = subprocess.Popen(
  343. args,
  344. cwd=self.gitroot,
  345. stdout=subprocess.PIPE,
  346. stderr=subprocess.PIPE,
  347. )
  348. if not check_for_daemon():
  349. raise SkipTest("git-daemon failed to start")
  350. def tearDown(self):
  351. with open(self.pidfile) as f:
  352. pid = int(f.read().strip())
  353. if sys.platform == "win32":
  354. PROCESS_TERMINATE = 1
  355. handle = ctypes.windll.kernel32.OpenProcess(PROCESS_TERMINATE, False, pid)
  356. ctypes.windll.kernel32.TerminateProcess(handle, -1)
  357. ctypes.windll.kernel32.CloseHandle(handle)
  358. else:
  359. with suppress(OSError):
  360. os.kill(pid, signal.SIGKILL)
  361. os.unlink(self.pidfile)
  362. self.process.wait()
  363. self.process.stdout.close()
  364. self.process.stderr.close()
  365. DulwichClientTestBase.tearDown(self)
  366. CompatTestCase.tearDown(self)
  367. def _client(self):
  368. return client.TCPGitClient("localhost")
  369. def _build_path(self, path):
  370. return path
  371. if sys.platform == "win32" and protocol.DEFAULT_GIT_PROTOCOL_VERSION_FETCH < 2:
  372. @expectedFailure
  373. def test_fetch_pack_no_side_band_64k(self):
  374. DulwichClientTestBase.test_fetch_pack_no_side_band_64k(self)
  375. def test_send_remove_branch(self):
  376. # This test fails intermittently on my machine, probably due to some sort
  377. # of race condition. Probably also related to #1015
  378. self.skipTest("skip flaky test; see #1015")
  379. @patch("dulwich.protocol.DEFAULT_GIT_PROTOCOL_VERSION_FETCH", new=0)
  380. class DulwichTCPClientTestGitProtov0(DulwichTCPClientTest):
  381. pass
  382. class TestSSHVendor:
  383. @staticmethod
  384. def run_command(
  385. host,
  386. command,
  387. username=None,
  388. port=None,
  389. password=None,
  390. key_filename=None,
  391. protocol_version=None,
  392. ):
  393. cmd, path = command.split(" ")
  394. cmd = cmd.split("-", 1)
  395. path = path.replace("'", "")
  396. env = dict(os.environ)
  397. if protocol_version is None:
  398. protocol_version = protocol.DEFAULT_GIT_PROTOCOL_VERSION_FETCH
  399. if protocol_version > 0:
  400. env["GIT_PROTOCOL"] = f"version={protocol_version}"
  401. p = subprocess.Popen(
  402. [*cmd, path],
  403. bufsize=0,
  404. stdin=subprocess.PIPE,
  405. stdout=subprocess.PIPE,
  406. stderr=subprocess.PIPE,
  407. env=env,
  408. )
  409. return client.SubprocessWrapper(p)
  410. class DulwichMockSSHClientTest(CompatTestCase, DulwichClientTestBase):
  411. def setUp(self):
  412. CompatTestCase.setUp(self)
  413. DulwichClientTestBase.setUp(self)
  414. self.real_vendor = client.get_ssh_vendor
  415. client.get_ssh_vendor = TestSSHVendor
  416. def tearDown(self):
  417. DulwichClientTestBase.tearDown(self)
  418. CompatTestCase.tearDown(self)
  419. client.get_ssh_vendor = self.real_vendor
  420. def _client(self):
  421. return client.SSHGitClient("localhost")
  422. def _build_path(self, path):
  423. return self.gitroot + path
  424. @patch("dulwich.protocol.DEFAULT_GIT_PROTOCOL_VERSION_FETCH", new=0)
  425. class DulwichMockSSHClientTestGitProtov0(DulwichMockSSHClientTest):
  426. pass
  427. class DulwichSubprocessClientTest(CompatTestCase, DulwichClientTestBase):
  428. def setUp(self):
  429. CompatTestCase.setUp(self)
  430. DulwichClientTestBase.setUp(self)
  431. def tearDown(self):
  432. DulwichClientTestBase.tearDown(self)
  433. CompatTestCase.tearDown(self)
  434. def _client(self):
  435. return client.SubprocessGitClient()
  436. def _build_path(self, path):
  437. return self.gitroot + path
  438. @patch("dulwich.protocol.DEFAULT_GIT_PROTOCOL_VERSION_FETCH", new=0)
  439. class DulwichSubprocessClientTestGitProtov0(DulwichSubprocessClientTest):
  440. pass
  441. class GitHTTPRequestHandler(http.server.SimpleHTTPRequestHandler):
  442. """HTTP Request handler that calls out to 'git http-backend'."""
  443. # Make rfile unbuffered -- we need to read one line and then pass
  444. # the rest to a subprocess, so we can't use buffered input.
  445. rbufsize = 0
  446. def do_POST(self):
  447. self.run_backend()
  448. def do_GET(self):
  449. self.run_backend()
  450. def send_head(self):
  451. return self.run_backend()
  452. def log_request(self, code="-", size="-"):
  453. # Let's be quiet, the test suite is noisy enough already
  454. pass
  455. def run_backend(self):
  456. """Call out to git http-backend."""
  457. # Based on CGIHTTPServer.CGIHTTPRequestHandler.run_cgi:
  458. # Copyright (c) 2001-2010 Python Software Foundation;
  459. # All Rights Reserved
  460. # Licensed under the Python Software Foundation License.
  461. rest = self.path
  462. # find an explicit query string, if present.
  463. i = rest.rfind("?")
  464. if i >= 0:
  465. rest, query = rest[:i], rest[i + 1 :]
  466. else:
  467. query = ""
  468. env = copy.deepcopy(os.environ)
  469. env["SERVER_SOFTWARE"] = self.version_string()
  470. env["SERVER_NAME"] = self.server.server_name
  471. env["GATEWAY_INTERFACE"] = "CGI/1.1"
  472. env["SERVER_PROTOCOL"] = self.protocol_version
  473. env["SERVER_PORT"] = str(self.server.server_port)
  474. env["GIT_PROJECT_ROOT"] = self.server.root_path
  475. env["GIT_HTTP_EXPORT_ALL"] = "1"
  476. env["REQUEST_METHOD"] = self.command
  477. uqrest = unquote(rest)
  478. env["PATH_INFO"] = uqrest
  479. env["SCRIPT_NAME"] = "/"
  480. if query:
  481. env["QUERY_STRING"] = query
  482. host = self.address_string()
  483. if host != self.client_address[0]:
  484. env["REMOTE_HOST"] = host
  485. env["REMOTE_ADDR"] = self.client_address[0]
  486. authorization = self.headers.get("authorization")
  487. if authorization:
  488. authorization = authorization.split()
  489. if len(authorization) == 2:
  490. import base64
  491. import binascii
  492. env["AUTH_TYPE"] = authorization[0]
  493. if authorization[0].lower() == "basic":
  494. try:
  495. authorization = base64.decodestring(authorization[1])
  496. except binascii.Error:
  497. pass
  498. else:
  499. authorization = authorization.split(":")
  500. if len(authorization) == 2:
  501. env["REMOTE_USER"] = authorization[0]
  502. # XXX REMOTE_IDENT
  503. content_type = self.headers.get("content-type")
  504. if content_type:
  505. env["CONTENT_TYPE"] = content_type
  506. length = self.headers.get("content-length")
  507. if length:
  508. env["CONTENT_LENGTH"] = length
  509. referer = self.headers.get("referer")
  510. if referer:
  511. env["HTTP_REFERER"] = referer
  512. accept = []
  513. for line in self.headers.getallmatchingheaders("accept"):
  514. if line[:1] in "\t\n\r ":
  515. accept.append(line.strip())
  516. else:
  517. accept = accept + line[7:].split(",")
  518. env["HTTP_ACCEPT"] = ",".join(accept)
  519. ua = self.headers.get("user-agent")
  520. if ua:
  521. env["HTTP_USER_AGENT"] = ua
  522. co = self.headers.get("cookie")
  523. if co:
  524. env["HTTP_COOKIE"] = co
  525. proto = self.headers.get("Git-Protocol")
  526. if proto:
  527. env["GIT_PROTOCOL"] = proto
  528. # XXX Other HTTP_* headers
  529. # Since we're setting the env in the parent, provide empty
  530. # values to override previously set values
  531. for k in (
  532. "QUERY_STRING",
  533. "REMOTE_HOST",
  534. "CONTENT_LENGTH",
  535. "HTTP_USER_AGENT",
  536. "HTTP_COOKIE",
  537. "HTTP_REFERER",
  538. ):
  539. env.setdefault(k, "")
  540. self.wfile.write(b"HTTP/1.1 200 Script output follows\r\n")
  541. self.wfile.write((f"Server: {self.server.server_name}\r\n").encode("ascii"))
  542. self.wfile.write((f"Date: {self.date_time_string()}\r\n").encode("ascii"))
  543. decoded_query = query.replace("+", " ")
  544. try:
  545. nbytes = int(length)
  546. except (TypeError, ValueError):
  547. nbytes = -1
  548. if self.command.lower() == "post":
  549. if nbytes > 0:
  550. data = self.rfile.read(nbytes)
  551. elif self.headers.get("transfer-encoding") == "chunked":
  552. chunks = []
  553. while True:
  554. line = self.rfile.readline()
  555. length = int(line.rstrip(), 16)
  556. chunk = self.rfile.read(length + 2)
  557. chunks.append(chunk[:-2])
  558. if length == 0:
  559. break
  560. data = b"".join(chunks)
  561. env["CONTENT_LENGTH"] = str(len(data))
  562. else:
  563. raise AssertionError
  564. else:
  565. data = None
  566. env["CONTENT_LENGTH"] = "0"
  567. # throw away additional data [see bug #427345]
  568. while select.select([self.rfile._sock], [], [], 0)[0]:
  569. if not self.rfile._sock.recv(1):
  570. break
  571. args = ["http-backend"]
  572. if "=" not in decoded_query:
  573. args.append(decoded_query)
  574. stdout = run_git_or_fail(args, input=data, env=env, stderr=subprocess.PIPE)
  575. self.wfile.write(stdout)
  576. class HTTPGitServer(http.server.HTTPServer):
  577. allow_reuse_address = True
  578. def __init__(self, server_address, root_path) -> None:
  579. http.server.HTTPServer.__init__(self, server_address, GitHTTPRequestHandler)
  580. self.root_path = root_path
  581. self.server_name = "localhost"
  582. def get_url(self):
  583. return f"http://{self.server_name}:{self.server_port}/"
  584. class DulwichHttpClientTest(CompatTestCase, DulwichClientTestBase):
  585. min_git_version = (1, 7, 0, 2)
  586. def setUp(self):
  587. CompatTestCase.setUp(self)
  588. DulwichClientTestBase.setUp(self)
  589. self._httpd = HTTPGitServer(("localhost", 0), self.gitroot)
  590. self.addCleanup(self._httpd.shutdown)
  591. threading.Thread(target=self._httpd.serve_forever).start()
  592. run_git_or_fail(["config", "http.uploadpack", "true"], cwd=self.dest)
  593. run_git_or_fail(["config", "http.receivepack", "true"], cwd=self.dest)
  594. def tearDown(self):
  595. DulwichClientTestBase.tearDown(self)
  596. CompatTestCase.tearDown(self)
  597. self._httpd.shutdown()
  598. self._httpd.socket.close()
  599. def _client(self):
  600. return client.HttpGitClient(self._httpd.get_url())
  601. def _build_path(self, path):
  602. return path
  603. def test_archive(self):
  604. raise SkipTest("exporting archives not supported over http")
  605. @patch("dulwich.protocol.DEFAULT_GIT_PROTOCOL_VERSION_FETCH", new=0)
  606. class DulwichHttpClientTestGitProtov0(DulwichHttpClientTest):
  607. pass