porcelain.py 27 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858
  1. # porcelain.py -- Porcelain-like layer on top of Dulwich
  2. # Copyright (C) 2013 Jelmer Vernooij <jelmer@samba.org>
  3. #
  4. # This program is free software; you can redistribute it and/or
  5. # modify it under the terms of the GNU General Public License
  6. # as published by the Free Software Foundation; either version 2
  7. # or (at your option) a later version of the License.
  8. #
  9. # This program is distributed in the hope that it will be useful,
  10. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  11. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  12. # GNU General Public License for more details.
  13. #
  14. # You should have received a copy of the GNU General Public License
  15. # along with this program; if not, write to the Free Software
  16. # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston,
  17. # MA 02110-1301, USA.
  18. """Simple wrapper that provides porcelain-like functions on top of Dulwich.
  19. Currently implemented:
  20. * archive
  21. * add
  22. * branch{_create,_delete,_list}
  23. * clone
  24. * commit
  25. * commit-tree
  26. * daemon
  27. * diff-tree
  28. * fetch
  29. * init
  30. * ls-remote
  31. * pull
  32. * push
  33. * rm
  34. * receive-pack
  35. * reset
  36. * rev-list
  37. * tag{_create,_delete,_list}
  38. * upload-pack
  39. * update-server-info
  40. * status
  41. * symbolic-ref
  42. These functions are meant to behave similarly to the git subcommands.
  43. Differences in behaviour are considered bugs.
  44. """
  45. __docformat__ = 'restructuredText'
  46. from collections import namedtuple
  47. from contextlib import (
  48. closing,
  49. contextmanager,
  50. )
  51. import os
  52. import sys
  53. import time
  54. from dulwich.archive import (
  55. tar_stream,
  56. )
  57. from dulwich.client import (
  58. get_transport_and_path,
  59. )
  60. from dulwich.errors import (
  61. SendPackError,
  62. UpdateRefsError,
  63. )
  64. from dulwich.index import get_unstaged_changes
  65. from dulwich.objects import (
  66. Commit,
  67. Tag,
  68. parse_timezone,
  69. )
  70. from dulwich.objectspec import (
  71. parse_object,
  72. parse_reftuples,
  73. )
  74. from dulwich.pack import (
  75. write_pack_index,
  76. write_pack_objects,
  77. )
  78. from dulwich.patch import write_tree_diff
  79. from dulwich.protocol import Protocol
  80. from dulwich.repo import (BaseRepo, Repo)
  81. from dulwich.server import (
  82. FileSystemBackend,
  83. TCPGitServer,
  84. ReceivePackHandler,
  85. UploadPackHandler,
  86. update_server_info as server_update_server_info,
  87. )
  88. # Module level tuple definition for status output
  89. GitStatus = namedtuple('GitStatus', 'staged unstaged untracked')
  90. default_bytes_out_stream = getattr(sys.stdout, 'buffer', sys.stdout)
  91. default_bytes_err_stream = getattr(sys.stderr, 'buffer', sys.stderr)
  92. def encode_path(path):
  93. """Encode a path as bytestring."""
  94. # TODO(jelmer): Use something other than ascii?
  95. if not isinstance(path, bytes):
  96. path = path.encode('ascii')
  97. return path
  98. def open_repo(path_or_repo):
  99. """Open an argument that can be a repository or a path for a repository."""
  100. if isinstance(path_or_repo, BaseRepo):
  101. return path_or_repo
  102. return Repo(path_or_repo)
  103. @contextmanager
  104. def _noop_context_manager(obj):
  105. """Context manager that has the same api as closing but does nothing."""
  106. yield obj
  107. def open_repo_closing(path_or_repo):
  108. """Open an argument that can be a repository or a path for a repository.
  109. returns a context manager that will close the repo on exit if the argument
  110. is a path, else does nothing if the argument is a repo.
  111. """
  112. if isinstance(path_or_repo, BaseRepo):
  113. return _noop_context_manager(path_or_repo)
  114. return closing(Repo(path_or_repo))
  115. def archive(repo, committish=None, outstream=sys.stdout,
  116. errstream=sys.stderr):
  117. """Create an archive.
  118. :param repo: Path of repository for which to generate an archive.
  119. :param committish: Commit SHA1 or ref to use
  120. :param outstream: Output stream (defaults to stdout)
  121. :param errstream: Error stream (defaults to stderr)
  122. """
  123. if committish is None:
  124. committish = "HEAD"
  125. with open_repo_closing(repo) as repo_obj:
  126. c = repo_obj[committish]
  127. tree = c.tree
  128. for chunk in tar_stream(repo_obj.object_store,
  129. repo_obj.object_store[c.tree], c.commit_time):
  130. outstream.write(chunk)
  131. def update_server_info(repo="."):
  132. """Update server info files for a repository.
  133. :param repo: path to the repository
  134. """
  135. with open_repo_closing(repo) as r:
  136. server_update_server_info(r)
  137. def symbolic_ref(repo, ref_name, force=False):
  138. """Set git symbolic ref into HEAD.
  139. :param repo: path to the repository
  140. :param ref_name: short name of the new ref
  141. :param force: force settings without checking if it exists in refs/heads
  142. """
  143. with open_repo_closing(repo) as repo_obj:
  144. ref_path = b'refs/heads/' + ref_name
  145. if not force and ref_path not in repo_obj.refs.keys():
  146. raise ValueError('fatal: ref `%s` is not a ref' % ref_name)
  147. repo_obj.refs.set_symbolic_ref(b'HEAD', ref_path)
  148. def commit(repo=".", message=None, author=None, committer=None):
  149. """Create a new commit.
  150. :param repo: Path to repository
  151. :param message: Optional commit message
  152. :param author: Optional author name and email
  153. :param committer: Optional committer name and email
  154. :return: SHA1 of the new commit
  155. """
  156. # FIXME: Support --all argument
  157. # FIXME: Support --signoff argument
  158. with open_repo_closing(repo) as r:
  159. return r.do_commit(message=message, author=author,
  160. committer=committer)
  161. def commit_tree(repo, tree, message=None, author=None, committer=None):
  162. """Create a new commit object.
  163. :param repo: Path to repository
  164. :param tree: An existing tree object
  165. :param author: Optional author name and email
  166. :param committer: Optional committer name and email
  167. """
  168. with open_repo_closing(repo) as r:
  169. return r.do_commit(message=message, tree=tree, committer=committer,
  170. author=author)
  171. def init(path=".", bare=False):
  172. """Create a new git repository.
  173. :param path: Path to repository.
  174. :param bare: Whether to create a bare repository.
  175. :return: A Repo instance
  176. """
  177. if not os.path.exists(path):
  178. os.mkdir(path)
  179. if bare:
  180. return Repo.init_bare(path)
  181. else:
  182. return Repo.init(path)
  183. def clone(source, target=None, bare=False, checkout=None, errstream=default_bytes_err_stream, outstream=None):
  184. """Clone a local or remote git repository.
  185. :param source: Path or URL for source repository
  186. :param target: Path to target repository (optional)
  187. :param bare: Whether or not to create a bare repository
  188. :param errstream: Optional stream to write progress to
  189. :param outstream: Optional stream to write progress to (deprecated)
  190. :return: The new repository
  191. """
  192. if outstream is not None:
  193. import warnings
  194. warnings.warn("outstream= has been deprecated in favour of errstream=.", DeprecationWarning,
  195. stacklevel=3)
  196. errstream = outstream
  197. if checkout is None:
  198. checkout = (not bare)
  199. if checkout and bare:
  200. raise ValueError("checkout and bare are incompatible")
  201. client, host_path = get_transport_and_path(source)
  202. if target is None:
  203. target = host_path.split("/")[-1]
  204. if not os.path.exists(target):
  205. os.mkdir(target)
  206. if bare:
  207. r = Repo.init_bare(target)
  208. else:
  209. r = Repo.init(target)
  210. try:
  211. remote_refs = client.fetch(host_path, r,
  212. determine_wants=r.object_store.determine_wants_all,
  213. progress=errstream.write)
  214. r[b"HEAD"] = remote_refs[b"HEAD"]
  215. if checkout:
  216. errstream.write(b'Checking out HEAD\n')
  217. r.reset_index()
  218. except:
  219. r.close()
  220. raise
  221. return r
  222. def add(repo=".", paths=None):
  223. """Add files to the staging area.
  224. :param repo: Repository for the files
  225. :param paths: Paths to add. No value passed stages all modified files.
  226. """
  227. # FIXME: Support patterns, directories.
  228. with open_repo_closing(repo) as r:
  229. if not paths:
  230. # If nothing is specified, add all non-ignored files.
  231. paths = []
  232. for dirpath, dirnames, filenames in os.walk(r.path):
  233. # Skip .git and below.
  234. if '.git' in dirnames:
  235. dirnames.remove('.git')
  236. for filename in filenames:
  237. paths.append(os.path.join(dirpath[len(r.path)+1:], filename))
  238. r.stage(paths)
  239. def rm(repo=".", paths=None):
  240. """Remove files from the staging area.
  241. :param repo: Repository for the files
  242. :param paths: Paths to remove
  243. """
  244. with open_repo_closing(repo) as r:
  245. index = r.open_index()
  246. for p in paths:
  247. del index[p.encode(sys.getfilesystemencoding())]
  248. index.write()
  249. def commit_decode(commit, contents, default_encoding='utf-8'):
  250. if commit.encoding is not None:
  251. return contents.decode(commit.encoding, "replace")
  252. return contents.decode(default_encoding, "replace")
  253. def print_commit(commit, decode, outstream=sys.stdout):
  254. """Write a human-readable commit log entry.
  255. :param commit: A `Commit` object
  256. :param outstream: A stream file to write to
  257. """
  258. outstream.write("-" * 50 + "\n")
  259. outstream.write("commit: " + commit.id.decode('ascii') + "\n")
  260. if len(commit.parents) > 1:
  261. outstream.write("merge: " +
  262. "...".join([c.decode('ascii') for c in commit.parents[1:]]) + "\n")
  263. outstream.write("author: " + decode(commit.author) + "\n")
  264. outstream.write("committer: " + decode(commit.committer) + "\n")
  265. outstream.write("\n")
  266. outstream.write(decode(commit.message) + "\n")
  267. outstream.write("\n")
  268. def print_tag(tag, decode, outstream=sys.stdout):
  269. """Write a human-readable tag.
  270. :param tag: A `Tag` object
  271. :param decode: Function for decoding bytes to unicode string
  272. :param outstream: A stream to write to
  273. """
  274. outstream.write("Tagger: " + decode(tag.tagger) + "\n")
  275. outstream.write("Date: " + decode(tag.tag_time) + "\n")
  276. outstream.write("\n")
  277. outstream.write(decode(tag.message) + "\n")
  278. outstream.write("\n")
  279. def show_blob(repo, blob, decode, outstream=sys.stdout):
  280. """Write a blob to a stream.
  281. :param repo: A `Repo` object
  282. :param blob: A `Blob` object
  283. :param decode: Function for decoding bytes to unicode string
  284. :param outstream: A stream file to write to
  285. """
  286. outstream.write(decode(blob.data))
  287. def show_commit(repo, commit, decode, outstream=sys.stdout):
  288. """Show a commit to a stream.
  289. :param repo: A `Repo` object
  290. :param commit: A `Commit` object
  291. :param decode: Function for decoding bytes to unicode string
  292. :param outstream: Stream to write to
  293. """
  294. print_commit(commit, decode=decode, outstream=outstream)
  295. parent_commit = repo[commit.parents[0]]
  296. write_tree_diff(outstream, repo.object_store, parent_commit.tree, commit.tree)
  297. def show_tree(repo, tree, decode, outstream=sys.stdout):
  298. """Print a tree to a stream.
  299. :param repo: A `Repo` object
  300. :param tree: A `Tree` object
  301. :param decode: Function for decoding bytes to unicode string
  302. :param outstream: Stream to write to
  303. """
  304. for n in tree:
  305. outstream.write(decode(n) + "\n")
  306. def show_tag(repo, tag, decode, outstream=sys.stdout):
  307. """Print a tag to a stream.
  308. :param repo: A `Repo` object
  309. :param tag: A `Tag` object
  310. :param decode: Function for decoding bytes to unicode string
  311. :param outstream: Stream to write to
  312. """
  313. print_tag(tag, decode, outstream)
  314. show_object(repo, repo[tag.object[1]], outstream)
  315. def show_object(repo, obj, decode, outstream):
  316. return {
  317. b"tree": show_tree,
  318. b"blob": show_blob,
  319. b"commit": show_commit,
  320. b"tag": show_tag,
  321. }[obj.type_name](repo, obj, decode, outstream)
  322. def log(repo=".", outstream=sys.stdout, max_entries=None):
  323. """Write commit logs.
  324. :param repo: Path to repository
  325. :param outstream: Stream to write log output to
  326. :param max_entries: Optional maximum number of entries to display
  327. """
  328. with open_repo_closing(repo) as r:
  329. walker = r.get_walker(max_entries=max_entries)
  330. for entry in walker:
  331. decode = lambda x: commit_decode(entry.commit, x)
  332. print_commit(entry.commit, decode, outstream)
  333. # TODO(jelmer): better default for encoding?
  334. def show(repo=".", objects=None, outstream=sys.stdout, default_encoding='utf-8'):
  335. """Print the changes in a commit.
  336. :param repo: Path to repository
  337. :param objects: Objects to show (defaults to [HEAD])
  338. :param outstream: Stream to write to
  339. :param default_encoding: Default encoding to use if none is set in the commit
  340. """
  341. if objects is None:
  342. objects = ["HEAD"]
  343. if not isinstance(objects, list):
  344. objects = [objects]
  345. with open_repo_closing(repo) as r:
  346. for objectish in objects:
  347. o = parse_object(r, objectish)
  348. if isinstance(o, Commit):
  349. decode = lambda x: commit_decode(o, x, default_encoding)
  350. else:
  351. decode = lambda x: x.decode(default_encoding)
  352. show_object(r, o, decode, outstream)
  353. def diff_tree(repo, old_tree, new_tree, outstream=sys.stdout):
  354. """Compares the content and mode of blobs found via two tree objects.
  355. :param repo: Path to repository
  356. :param old_tree: Id of old tree
  357. :param new_tree: Id of new tree
  358. :param outstream: Stream to write to
  359. """
  360. with open_repo_closing(repo) as r:
  361. write_tree_diff(outstream, r.object_store, old_tree, new_tree)
  362. def rev_list(repo, commits, outstream=sys.stdout):
  363. """Lists commit objects in reverse chronological order.
  364. :param repo: Path to repository
  365. :param commits: Commits over which to iterate
  366. :param outstream: Stream to write to
  367. """
  368. with open_repo_closing(repo) as r:
  369. for entry in r.get_walker(include=[r[c].id for c in commits]):
  370. outstream.write(entry.commit.id + b"\n")
  371. def tag(*args, **kwargs):
  372. import warnings
  373. warnings.warn("tag has been deprecated in favour of tag_create.", DeprecationWarning)
  374. return tag_create(*args, **kwargs)
  375. def tag_create(repo, tag, author=None, message=None, annotated=False,
  376. objectish="HEAD", tag_time=None, tag_timezone=None):
  377. """Creates a tag in git via dulwich calls:
  378. :param repo: Path to repository
  379. :param tag: tag string
  380. :param author: tag author (optional, if annotated is set)
  381. :param message: tag message (optional)
  382. :param annotated: whether to create an annotated tag
  383. :param objectish: object the tag should point at, defaults to HEAD
  384. :param tag_time: Optional time for annotated tag
  385. :param tag_timezone: Optional timezone for annotated tag
  386. """
  387. with open_repo_closing(repo) as r:
  388. object = parse_object(r, objectish)
  389. if annotated:
  390. # Create the tag object
  391. tag_obj = Tag()
  392. if author is None:
  393. # TODO(jelmer): Don't use repo private method.
  394. author = r._get_user_identity()
  395. tag_obj.tagger = author
  396. tag_obj.message = message
  397. tag_obj.name = tag
  398. tag_obj.object = (type(object), object.id)
  399. tag_obj.tag_time = tag_time
  400. if tag_time is None:
  401. tag_time = int(time.time())
  402. if tag_timezone is None:
  403. # TODO(jelmer) Use current user timezone rather than UTC
  404. tag_timezone = 0
  405. elif isinstance(tag_timezone, str):
  406. tag_timezone = parse_timezone(tag_timezone)
  407. tag_obj.tag_timezone = tag_timezone
  408. r.object_store.add_object(tag_obj)
  409. tag_id = tag_obj.id
  410. else:
  411. tag_id = object.id
  412. r.refs[b'refs/tags/' + tag] = tag_id
  413. def list_tags(*args, **kwargs):
  414. import warnings
  415. warnings.warn("list_tags has been deprecated in favour of tag_list.", DeprecationWarning)
  416. return tag_list(*args, **kwargs)
  417. def tag_list(repo, outstream=sys.stdout):
  418. """List all tags.
  419. :param repo: Path to repository
  420. :param outstream: Stream to write tags to
  421. """
  422. with open_repo_closing(repo) as r:
  423. tags = list(r.refs.as_dict(b"refs/tags"))
  424. tags.sort()
  425. return tags
  426. def tag_delete(repo, name):
  427. """Remove a tag.
  428. :param repo: Path to repository
  429. :param name: Name of tag to remove
  430. """
  431. with open_repo_closing(repo) as r:
  432. if isinstance(name, bytes):
  433. names = [name]
  434. elif isinstance(name, list):
  435. names = name
  436. else:
  437. raise TypeError("Unexpected tag name type %r" % name)
  438. for name in names:
  439. del r.refs[b"refs/tags/" + name]
  440. def reset(repo, mode, committish="HEAD"):
  441. """Reset current HEAD to the specified state.
  442. :param repo: Path to repository
  443. :param mode: Mode ("hard", "soft", "mixed")
  444. """
  445. if mode != "hard":
  446. raise ValueError("hard is the only mode currently supported")
  447. with open_repo_closing(repo) as r:
  448. tree = r[committish].tree
  449. r.reset_index()
  450. def push(repo, remote_location, refspecs=None,
  451. outstream=sys.stdout, errstream=sys.stderr):
  452. """Remote push with dulwich via dulwich.client
  453. :param repo: Path to repository
  454. :param remote_location: Location of the remote
  455. :param refspecs: relative path to the refs to push to remote
  456. :param outstream: A stream file to write output
  457. :param errstream: A stream file to write errors
  458. """
  459. # Open the repo
  460. with open_repo_closing(repo) as r:
  461. # Get the client and path
  462. client, path = get_transport_and_path(remote_location)
  463. selected_refs = []
  464. def update_refs(refs):
  465. selected_refs.extend(parse_reftuples(r.refs, refs, refspecs))
  466. # TODO: Handle selected_refs == {None: None}
  467. for (lh, rh, force) in selected_refs:
  468. if lh is None:
  469. del refs[rh]
  470. else:
  471. refs[rh] = r.refs[lh]
  472. return refs
  473. err_encoding = getattr(errstream, 'encoding', 'utf-8')
  474. remote_location_bytes = remote_location.encode(err_encoding)
  475. try:
  476. client.send_pack(path, update_refs,
  477. r.object_store.generate_pack_contents, progress=errstream.write)
  478. errstream.write(b"Push to " + remote_location_bytes +
  479. b" successful.\n")
  480. except (UpdateRefsError, SendPackError) as e:
  481. errstream.write(b"Push to " + remote_location_bytes +
  482. b" failed -> " + e.message.encode(err_encoding) +
  483. b"\n")
  484. def pull(repo, remote_location, refspecs=None,
  485. outstream=sys.stdout, errstream=sys.stderr):
  486. """Pull from remote via dulwich.client
  487. :param repo: Path to repository
  488. :param remote_location: Location of the remote
  489. :param refspec: refspecs to fetch
  490. :param outstream: A stream file to write to output
  491. :param errstream: A stream file to write to errors
  492. """
  493. # Open the repo
  494. with open_repo_closing(repo) as r:
  495. selected_refs = []
  496. def determine_wants(remote_refs):
  497. selected_refs.extend(parse_reftuples(remote_refs, r.refs, refspecs))
  498. return [remote_refs[lh] for (lh, rh, force) in selected_refs]
  499. client, path = get_transport_and_path(remote_location)
  500. remote_refs = client.fetch(path, r, progress=errstream.write,
  501. determine_wants=determine_wants)
  502. for (lh, rh, force) in selected_refs:
  503. r.refs[rh] = remote_refs[lh]
  504. if selected_refs:
  505. r[b'HEAD'] = remote_refs[selected_refs[0][1]]
  506. # Perform 'git checkout .' - syncs staged changes
  507. tree = r[b"HEAD"].tree
  508. r.reset_index()
  509. def status(repo="."):
  510. """Returns staged, unstaged, and untracked changes relative to the HEAD.
  511. :param repo: Path to repository or repository object
  512. :return: GitStatus tuple,
  513. staged - list of staged paths (diff index/HEAD)
  514. unstaged - list of unstaged paths (diff index/working-tree)
  515. untracked - list of untracked, un-ignored & non-.git paths
  516. """
  517. with open_repo_closing(repo) as r:
  518. # 1. Get status of staged
  519. tracked_changes = get_tree_changes(r)
  520. # 2. Get status of unstaged
  521. unstaged_changes = list(get_unstaged_changes(r.open_index(), r.path))
  522. # TODO - Status of untracked - add untracked changes, need gitignore.
  523. untracked_changes = []
  524. return GitStatus(tracked_changes, unstaged_changes, untracked_changes)
  525. def get_tree_changes(repo):
  526. """Return add/delete/modify changes to tree by comparing index to HEAD.
  527. :param repo: repo path or object
  528. :return: dict with lists for each type of change
  529. """
  530. with open_repo_closing(repo) as r:
  531. index = r.open_index()
  532. # Compares the Index to the HEAD & determines changes
  533. # Iterate through the changes and report add/delete/modify
  534. # TODO: call out to dulwich.diff_tree somehow.
  535. tracked_changes = {
  536. 'add': [],
  537. 'delete': [],
  538. 'modify': [],
  539. }
  540. try:
  541. tree_id = r[b'HEAD'].tree
  542. except KeyError:
  543. tree_id = None
  544. for change in index.changes_from_tree(r.object_store, tree_id):
  545. if not change[0][0]:
  546. tracked_changes['add'].append(change[0][1])
  547. elif not change[0][1]:
  548. tracked_changes['delete'].append(change[0][0])
  549. elif change[0][0] == change[0][1]:
  550. tracked_changes['modify'].append(change[0][0])
  551. else:
  552. raise AssertionError('git mv ops not yet supported')
  553. return tracked_changes
  554. def daemon(path=".", address=None, port=None):
  555. """Run a daemon serving Git requests over TCP/IP.
  556. :param path: Path to the directory to serve.
  557. :param address: Optional address to listen on (defaults to ::)
  558. :param port: Optional port to listen on (defaults to TCP_GIT_PORT)
  559. """
  560. # TODO(jelmer): Support git-daemon-export-ok and --export-all.
  561. backend = FileSystemBackend(path)
  562. server = TCPGitServer(backend, address, port)
  563. server.serve_forever()
  564. def web_daemon(path=".", address=None, port=None):
  565. """Run a daemon serving Git requests over HTTP.
  566. :param path: Path to the directory to serve
  567. :param address: Optional address to listen on (defaults to ::)
  568. :param port: Optional port to listen on (defaults to 80)
  569. """
  570. from dulwich.web import (
  571. make_wsgi_chain,
  572. make_server,
  573. WSGIRequestHandlerLogger,
  574. WSGIServerLogger)
  575. backend = FileSystemBackend(path)
  576. app = make_wsgi_chain(backend)
  577. server = make_server(address, port, app,
  578. handler_class=WSGIRequestHandlerLogger,
  579. server_class=WSGIServerLogger)
  580. server.serve_forever()
  581. def upload_pack(path=".", inf=None, outf=None):
  582. """Upload a pack file after negotiating its contents using smart protocol.
  583. :param path: Path to the repository
  584. :param inf: Input stream to communicate with client
  585. :param outf: Output stream to communicate with client
  586. """
  587. if outf is None:
  588. outf = getattr(sys.stdout, 'buffer', sys.stdout)
  589. if inf is None:
  590. inf = getattr(sys.stdin, 'buffer', sys.stdin)
  591. backend = FileSystemBackend(path)
  592. def send_fn(data):
  593. outf.write(data)
  594. outf.flush()
  595. proto = Protocol(inf.read, send_fn)
  596. handler = UploadPackHandler(backend, [path], proto)
  597. # FIXME: Catch exceptions and write a single-line summary to outf.
  598. handler.handle()
  599. return 0
  600. def receive_pack(path=".", inf=None, outf=None):
  601. """Receive a pack file after negotiating its contents using smart protocol.
  602. :param path: Path to the repository
  603. :param inf: Input stream to communicate with client
  604. :param outf: Output stream to communicate with client
  605. """
  606. if outf is None:
  607. outf = getattr(sys.stdout, 'buffer', sys.stdout)
  608. if inf is None:
  609. inf = getattr(sys.stdin, 'buffer', sys.stdin)
  610. backend = FileSystemBackend(path)
  611. def send_fn(data):
  612. outf.write(data)
  613. outf.flush()
  614. proto = Protocol(inf.read, send_fn)
  615. handler = ReceivePackHandler(backend, [path], proto)
  616. # FIXME: Catch exceptions and write a single-line summary to outf.
  617. handler.handle()
  618. return 0
  619. def branch_delete(repo, name):
  620. """Delete a branch.
  621. :param repo: Path to the repository
  622. :param name: Name of the branch
  623. """
  624. with open_repo_closing(repo) as r:
  625. if isinstance(name, bytes):
  626. names = [name]
  627. elif isinstance(name, list):
  628. names = name
  629. else:
  630. raise TypeError("Unexpected branch name type %r" % name)
  631. for name in names:
  632. del r.refs[b"refs/heads/" + name]
  633. def branch_create(repo, name, objectish=None, force=False):
  634. """Create a branch.
  635. :param repo: Path to the repository
  636. :param name: Name of the new branch
  637. :param objectish: Target object to point new branch at (defaults to HEAD)
  638. :param force: Force creation of branch, even if it already exists
  639. """
  640. with open_repo_closing(repo) as r:
  641. if isinstance(name, bytes):
  642. names = [name]
  643. elif isinstance(name, list):
  644. names = name
  645. else:
  646. raise TypeError("Unexpected branch name type %r" % name)
  647. if objectish is None:
  648. objectish = "HEAD"
  649. object = parse_object(r, objectish)
  650. refname = b"refs/heads/" + name
  651. if refname in r.refs and not force:
  652. raise KeyError("Branch with name %s already exists." % name)
  653. r.refs[refname] = object.id
  654. def branch_list(repo):
  655. """List all branches.
  656. :param repo: Path to the repository
  657. """
  658. with open_repo_closing(repo) as r:
  659. return r.refs.keys(base=b"refs/heads/")
  660. def fetch(repo, remote_location, outstream=sys.stdout, errstream=sys.stderr):
  661. """Fetch objects from a remote server.
  662. :param repo: Path to the repository
  663. :param remote_location: String identifying a remote server
  664. :param outstream: Output stream (defaults to stdout)
  665. :param errstream: Error stream (defaults to stderr)
  666. :return: Dictionary with refs on the remote
  667. """
  668. with open_repo_closing(repo) as r:
  669. client, path = get_transport_and_path(remote_location)
  670. remote_refs = client.fetch(path, r, progress=errstream.write)
  671. return remote_refs
  672. def ls_remote(remote):
  673. client, host_path = get_transport_and_path(remote)
  674. return client.get_refs(encode_path(host_path))
  675. def repack(repo):
  676. """Repack loose files in a repository.
  677. Currently this only packs loose objects.
  678. :param repo: Path to the repository
  679. """
  680. with open_repo_closing(repo) as r:
  681. r.object_store.pack_loose_objects()
  682. def pack_objects(repo, object_ids, packf, idxf, delta_window_size=None):
  683. """Pack objects into a file.
  684. :param repo: Path to the repository
  685. :param object_ids: List of object ids to write
  686. :param packf: File-like object to write to
  687. :param idxf: File-like object to write to (can be None)
  688. """
  689. with open_repo_closing(repo) as r:
  690. entries, data_sum = write_pack_objects(
  691. packf,
  692. r.object_store.iter_shas((oid, None) for oid in object_ids),
  693. delta_window_size=delta_window_size)
  694. if idxf is not None:
  695. entries = [(k, v[0], v[1]) for (k, v) in entries.items()]
  696. entries.sort()
  697. write_pack_index(idxf, entries, data_sum)