3 # dulwich - Simple command-line interface to Dulwich
4 # Copyright (C) 2008-2011 Jelmer Vernooij <jelmer@jelmer.uk>
7 # Dulwich is dual-licensed under the Apache License, Version 2.0 and the GNU
8 # General Public License as public by the Free Software Foundation; version 2.0
9 # or (at your option) any later version. You can redistribute it and/or
10 # modify it under the terms of either of these two licenses.
12 # Unless required by applicable law or agreed to in writing, software
13 # distributed under the License is distributed on an "AS IS" BASIS,
14 # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15 # See the License for the specific language governing permissions and
16 # limitations under the License.
18 # You should have received a copy of the licenses; if not, see
19 # <http://www.gnu.org/licenses/> for a copy of the GNU General Public License
20 # and <http://www.apache.org/licenses/LICENSE-2.0> for a copy of the Apache
21 # License, Version 2.0.
24 """Simple command-line interface to Dulwich>
26 This is a very simple command-line wrapper for Dulwich. It is by
27 no means intended to be a full-blown Git command-line interface but just
28 a way to test Dulwich.
33 from getopt import getopt
37 def signal_int(signal, frame):
41 def signal_quit(signal, frame):
45 if 'DULWICH_PDB' in os.environ:
46 signal.signal(signal.SIGQUIT, signal_quit)
47 signal.signal(signal.SIGINT, signal_int)
49 from dulwich import porcelain
50 from dulwich.client import get_transport_and_path
51 from dulwich.errors import ApplyDeltaError
52 from dulwich.index import Index
53 from dulwich.pack import Pack, sha_to_hex
54 from dulwich.patch import write_tree_diff
55 from dulwich.repo import Repo
58 class Command(object):
59 """A Dulwich subcommand."""
62 """Run the command."""
63 raise NotImplementedError(self.run)
66 class cmd_archive(Command):
69 parser = optparse.OptionParser()
70 parser.add_option("--remote", type=str,
71 help="Retrieve archive from specified remote repo")
72 options, args = parser.parse_args(args)
73 committish = args.pop(0)
75 client, path = get_transport_and_path(options.remote)
76 client.archive(path, committish, sys.stdout.write,
77 write_error=sys.stderr.write)
79 porcelain.archive('.', committish, outstream=sys.stdout,
83 class cmd_add(Command):
86 opts, args = getopt(args, "", [])
88 porcelain.add(".", paths=args)
91 class cmd_rm(Command):
94 opts, args = getopt(args, "", [])
96 porcelain.rm(".", paths=args)
99 class cmd_fetch_pack(Command):
102 opts, args = getopt(args, "", ["all"])
104 client, path = get_transport_and_path(args.pop(0))
107 determine_wants = r.object_store.determine_wants_all
109 determine_wants = lambda x: [y for y in args if not y in r.object_store]
110 client.fetch(path, r, determine_wants)
113 class cmd_fetch(Command):
116 opts, args = getopt(args, "", [])
118 client, path = get_transport_and_path(args.pop(0))
121 determine_wants = r.object_store.determine_wants_all
122 refs = client.fetch(path, r, progress=sys.stdout.write)
123 print("Remote refs:")
124 for item in refs.items():
125 print("%s -> %s" % item)
128 class cmd_fsck(Command):
131 opts, args = getopt(args, "", [])
133 for (obj, msg) in porcelain.fsck('.'):
134 print("%s: %s" % (obj, msg))
137 class cmd_log(Command):
140 parser = optparse.OptionParser()
141 parser.add_option("--reverse", dest="reverse", action="store_true",
142 help="Reverse order in which entries are printed")
143 parser.add_option("--name-status", dest="name_status", action="store_true",
144 help="Print name/status for each changed file")
145 options, args = parser.parse_args(args)
147 porcelain.log(".", paths=args, reverse=options.reverse,
148 name_status=options.name_status,
149 outstream=sys.stdout)
152 class cmd_diff(Command):
155 opts, args = getopt(args, "", [])
158 print("Usage: dulwich diff COMMITID")
163 commit = r[commit_id]
164 parent_commit = r[commit.parents[0]]
165 write_tree_diff(sys.stdout, r.object_store, parent_commit.tree, commit.tree)
168 class cmd_dump_pack(Command):
171 opts, args = getopt(args, "", [])
174 print("Usage: dulwich dump-pack FILENAME")
177 basename, _ = os.path.splitext(args[0])
179 print("Object names checksum: %s" % x.name())
180 print("Checksum: %s" % sha_to_hex(x.get_stored_checksum()))
182 print("CHECKSUM DOES NOT MATCH")
183 print("Length: %d" % len(x))
186 print("\t%s" % x[name])
187 except KeyError as k:
188 print("\t%s: Unable to resolve base %s" % (name, k))
189 except ApplyDeltaError as e:
190 print("\t%s: Unable to apply delta: %r" % (name, e))
193 class cmd_dump_index(Command):
196 opts, args = getopt(args, "", [])
199 print("Usage: dulwich dump-index FILENAME")
203 idx = Index(filename)
209 class cmd_init(Command):
212 opts, args = getopt(args, "", ["bare"])
220 porcelain.init(path, bare=("--bare" in opts))
223 class cmd_clone(Command):
226 parser = optparse.OptionParser()
227 parser.add_option("--bare", dest="bare",
228 help="Whether to create a bare repository.",
230 parser.add_option("--depth", dest="depth",
231 type=int, help="Depth at which to fetch")
232 options, args = parser.parse_args(args)
235 print("usage: dulwich clone host:path [PATH]")
244 porcelain.clone(source, target, bare=options.bare, depth=options.depth)
247 class cmd_commit(Command):
250 opts, args = getopt(args, "", ["message"])
252 porcelain.commit(".", message=opts["--message"])
255 class cmd_commit_tree(Command):
258 opts, args = getopt(args, "", ["message"])
260 print("usage: dulwich commit-tree tree")
263 porcelain.commit_tree(".", tree=args[0], message=opts["--message"])
266 class cmd_update_server_info(Command):
269 porcelain.update_server_info(".")
272 class cmd_symbolic_ref(Command):
275 opts, args = getopt(args, "", ["ref-name", "force"])
277 print("Usage: dulwich symbolic-ref REF_NAME [--force]")
280 ref_name = args.pop(0)
281 porcelain.symbolic_ref(".", ref_name=ref_name, force='--force' in args)
284 class cmd_show(Command):
287 opts, args = getopt(args, "", [])
288 porcelain.show(".", args)
291 class cmd_diff_tree(Command):
294 opts, args = getopt(args, "", [])
296 print("Usage: dulwich diff-tree OLD-TREE NEW-TREE")
298 porcelain.diff_tree(".", args[0], args[1])
301 class cmd_rev_list(Command):
304 opts, args = getopt(args, "", [])
306 print('Usage: dulwich rev-list COMMITID...')
308 porcelain.rev_list('.', args)
311 class cmd_tag(Command):
314 parser = optparse.OptionParser()
315 parser.add_option("-a", "--annotated", help="Create an annotated tag.", action="store_true")
316 parser.add_option("-s", "--sign", help="Sign the annotated tag.", action="store_true")
317 options, args = parser.parse_args(args)
318 porcelain.tag_create(
319 '.', args[0], annotated=options.annotated,
323 class cmd_repack(Command):
326 opts, args = getopt(args, "", [])
328 porcelain.repack('.')
331 class cmd_reset(Command):
334 opts, args = getopt(args, "", ["hard", "soft", "mixed"])
339 elif "--soft" in opts:
341 elif "--mixed" in opts:
343 porcelain.reset('.', mode=mode, *args)
346 class cmd_daemon(Command):
349 from dulwich import log_utils
350 from dulwich.protocol import TCP_GIT_PORT
351 parser = optparse.OptionParser()
352 parser.add_option("-l", "--listen_address", dest="listen_address",
354 help="Binding IP address.")
355 parser.add_option("-p", "--port", dest="port", type=int,
356 default=TCP_GIT_PORT,
357 help="Binding TCP port.")
358 options, args = parser.parse_args(args)
360 log_utils.default_logging_config()
365 from dulwich import porcelain
366 porcelain.daemon(gitdir, address=options.listen_address,
370 class cmd_web_daemon(Command):
373 from dulwich import log_utils
374 parser = optparse.OptionParser()
375 parser.add_option("-l", "--listen_address", dest="listen_address",
377 help="Binding IP address.")
378 parser.add_option("-p", "--port", dest="port", type=int,
380 help="Binding TCP port.")
381 options, args = parser.parse_args(args)
383 log_utils.default_logging_config()
388 from dulwich import porcelain
389 porcelain.web_daemon(gitdir, address=options.listen_address,
393 class cmd_write_tree(Command):
396 parser = optparse.OptionParser()
397 options, args = parser.parse_args(args)
398 sys.stdout.write('%s\n' % porcelain.write_tree('.'))
401 class cmd_receive_pack(Command):
404 parser = optparse.OptionParser()
405 options, args = parser.parse_args(args)
410 porcelain.receive_pack(gitdir)
413 class cmd_upload_pack(Command):
416 parser = optparse.OptionParser()
417 options, args = parser.parse_args(args)
422 porcelain.upload_pack(gitdir)
425 class cmd_status(Command):
428 parser = optparse.OptionParser()
429 options, args = parser.parse_args(args)
434 status = porcelain.status(gitdir)
435 if any(names for (kind, names) in status.staged.items()):
436 sys.stdout.write("Changes to be committed:\n\n")
437 for kind, names in status.staged.items():
439 sys.stdout.write("\t%s: %s\n" % (
440 kind, name.decode(sys.getfilesystemencoding())))
441 sys.stdout.write("\n")
443 sys.stdout.write("Changes not staged for commit:\n\n")
444 for name in status.unstaged:
445 sys.stdout.write("\t%s\n" %
446 name.decode(sys.getfilesystemencoding()))
447 sys.stdout.write("\n")
449 sys.stdout.write("Untracked files:\n\n")
450 for name in status.untracked:
451 sys.stdout.write("\t%s\n" % name)
452 sys.stdout.write("\n")
455 class cmd_ls_remote(Command):
458 opts, args = getopt(args, '', [])
460 print('Usage: dulwich ls-remote URL')
462 refs = porcelain.ls_remote(args[0])
463 for ref in sorted(refs):
464 sys.stdout.write("%s\t%s\n" % (ref, refs[ref]))
467 class cmd_ls_tree(Command):
470 parser = optparse.OptionParser()
471 parser.add_option("-r", "--recursive", action="store_true",
472 help="Recusively list tree contents.")
473 parser.add_option("--name-only", action="store_true",
474 help="Only display name.")
475 options, args = parser.parse_args(args)
477 treeish = args.pop(0)
481 '.', treeish, outstream=sys.stdout, recursive=options.recursive,
482 name_only=options.name_only)
485 class cmd_pack_objects(Command):
488 opts, args = getopt(args, '', ['stdout'])
490 if len(args) < 1 and not '--stdout' in args:
491 print('Usage: dulwich pack-objects basename')
493 object_ids = [l.strip() for l in sys.stdin.readlines()]
495 if '--stdout' in opts:
496 packf = getattr(sys.stdout, 'buffer', sys.stdout)
500 packf = open(basename + '.pack', 'w')
501 idxf = open(basename + '.idx', 'w')
502 close = [packf, idxf]
503 porcelain.pack_objects('.', object_ids, packf, idxf)
508 class cmd_pull(Command):
511 parser = optparse.OptionParser()
512 options, args = parser.parse_args(args)
514 from_location = args[0]
517 porcelain.pull('.', from_location)
520 class cmd_push(Command):
523 parser = optparse.OptionParser()
524 options, args = parser.parse_args(args)
526 print("Usage: dulwich push TO-LOCATION REFSPEC..")
528 to_location = args[0]
530 porcelain.push('.', to_location, refspecs)
533 class cmd_remote_add(Command):
536 parser = optparse.OptionParser()
537 options, args = parser.parse_args(args)
538 porcelain.remote_add('.', args[0], args[1])
541 class SuperCommand(Command):
547 print("Supported subcommands: %s" % ', '.join(self.subcommands.keys()))
551 cmd_kls = self.subcommands[cmd]
553 print('No such subcommand: %s' % args[0])
555 return cmd_kls().run(args[1:])
558 class cmd_remote(SuperCommand):
561 "add": cmd_remote_add,
565 class cmd_check_ignore(Command):
568 parser = optparse.OptionParser()
569 options, args = parser.parse_args(args)
571 for path in porcelain.check_ignore('.', args):
577 class cmd_check_mailmap(Command):
580 parser = optparse.OptionParser()
581 options, args = parser.parse_args(args)
583 canonical_identity = porcelain.check_mailmap('.', arg)
584 print(canonical_identity)
587 class cmd_stash_list(Command):
590 parser = optparse.OptionParser()
591 options, args = parser.parse_args(args)
592 for i, entry in porcelain.stash_list('.'):
593 print("stash@{%d}: %s" % (i, entry.message.rstrip('\n')))
596 class cmd_stash_push(Command):
599 parser = optparse.OptionParser()
600 options, args = parser.parse_args(args)
601 porcelain.stash_push('.')
602 print("Saved working directory and index state")
605 class cmd_stash_pop(Command):
608 parser = optparse.OptionParser()
609 options, args = parser.parse_args(args)
610 porcelain.stash_pop('.')
611 print("Restrored working directory and index state")
614 class cmd_stash(SuperCommand):
617 "list": cmd_stash_list,
618 "pop": cmd_stash_pop,
619 "push": cmd_stash_push,
623 class cmd_ls_files(Command):
626 parser = optparse.OptionParser()
627 options, args = parser.parse_args(args)
628 for name in porcelain.ls_files('.'):
632 class cmd_describe(Command):
635 parser = optparse.OptionParser()
636 options, args = parser.parse_args(args)
637 print(porcelain.describe('.'))
640 class cmd_help(Command):
643 parser = optparse.OptionParser()
644 parser.add_option("-a", "--all", dest="all",
646 help="List all commands.")
647 options, args = parser.parse_args(args)
650 print('Available commands:')
651 for cmd in sorted(commands):
655 The dulwich command line tool is currently a very basic frontend for the
656 Dulwich python module. For full functionality, please see the API reference.
658 For a list of supported commands, see 'dulwich help -a'.
664 "archive": cmd_archive,
665 "check-ignore": cmd_check_ignore,
666 "check-mailmap": cmd_check_mailmap,
668 "commit": cmd_commit,
669 "commit-tree": cmd_commit_tree,
670 "describe": cmd_describe,
671 "daemon": cmd_daemon,
673 "diff-tree": cmd_diff_tree,
674 "dump-pack": cmd_dump_pack,
675 "dump-index": cmd_dump_index,
676 "fetch-pack": cmd_fetch_pack,
682 "ls-files": cmd_ls_files,
683 "ls-remote": cmd_ls_remote,
684 "ls-tree": cmd_ls_tree,
685 "pack-objects": cmd_pack_objects,
688 "receive-pack": cmd_receive_pack,
689 "remote": cmd_remote,
690 "repack": cmd_repack,
692 "rev-list": cmd_rev_list,
696 "status": cmd_status,
697 "symbolic-ref": cmd_symbolic_ref,
699 "update-server-info": cmd_update_server_info,
700 "upload-pack": cmd_upload_pack,
701 "web-daemon": cmd_web_daemon,
702 "write-tree": cmd_write_tree,
705 if len(sys.argv) < 2:
706 print("Usage: %s <%s> [OPTIONS...]" % (sys.argv[0], "|".join(commands.keys())))
711 cmd_kls = commands[cmd]
713 print("No such subcommand: %s" % cmd)
715 # TODO(jelmer): Return non-0 on errors
716 cmd_kls().run(sys.argv[2:])