]> arthur.barton.de Git - bup.git/blob - lib/cmd/tag-cmd.py
Remove $(dirname "$0") from sys.path
[bup.git] / lib / cmd / tag-cmd.py
1 #!/bin/sh
2 """": # -*-python-*-
3 # https://sourceware.org/bugzilla/show_bug.cgi?id=26034
4 export "BUP_ARGV_0"="$0"
5 arg_i=1
6 for arg in "$@"; do
7     export "BUP_ARGV_${arg_i}"="$arg"
8     shift
9     arg_i=$((arg_i + 1))
10 done
11 # Here to end of preamble replaced during install
12 bup_python="$(dirname "$0")/../../config/bin/python" || exit $?
13 exec "$bup_python" "$0"
14 """
15 # end of bup preamble
16
17 from __future__ import absolute_import
18
19 # Intentionally replace the dirname "$0" that python prepends
20 import os, sys
21 sys.path[0] = os.path.dirname(os.path.realpath(__file__)) + '/..'
22
23 from binascii import hexlify
24
25 from bup import compat, git, options
26 from bup.compat import argv_bytes
27 from bup.helpers import debug1, handle_ctrl_c, log
28 from bup.io import byte_stream, path_msg
29
30 # FIXME: review for safe writes.
31
32 handle_ctrl_c()
33
34 optspec = """
35 bup tag
36 bup tag [-f] <tag name> <commit>
37 bup tag [-f] -d <tag name>
38 --
39 d,delete=   Delete a tag
40 f,force     Overwrite existing tag, or ignore missing tag when deleting
41 """
42
43 o = options.Options(optspec)
44 opt, flags, extra = o.parse(compat.argv[1:])
45
46 git.check_repo_or_die()
47
48 tags = [t for sublist in git.tags().values() for t in sublist]
49
50 if opt.delete:
51     # git.delete_ref() doesn't complain if a ref doesn't exist.  We
52     # could implement this verification but we'd need to read in the
53     # contents of the tag file and pass the hash, and we already know
54     # about the tag's existance via "tags".
55     tag_name = argv_bytes(opt.delete)
56     if not opt.force and tag_name not in tags:
57         log("error: tag '%s' doesn't exist\n" % path_msg(tag_name))
58         sys.exit(1)
59     tag_file = b'refs/tags/%s' % tag_name
60     git.delete_ref(tag_file)
61     sys.exit(0)
62
63 if not extra:
64     for t in tags:
65         sys.stdout.flush()
66         out = byte_stream(sys.stdout)
67         out.write(t)
68         out.write(b'\n')
69     sys.exit(0)
70 elif len(extra) != 2:
71     o.fatal('expected commit ref and hash')
72
73 tag_name, commit = map(argv_bytes, extra[:2])
74 if not tag_name:
75     o.fatal("tag name must not be empty.")
76 debug1("args: tag name = %s; commit = %s\n"
77        % (path_msg(tag_name), commit.decode('ascii')))
78
79 if tag_name in tags and not opt.force:
80     log("bup: error: tag '%s' already exists\n" % path_msg(tag_name))
81     sys.exit(1)
82
83 if tag_name.startswith(b'.'):
84     o.fatal("'%s' is not a valid tag name." % path_msg(tag_name))
85
86 try:
87     hash = git.rev_parse(commit)
88 except git.GitError as e:
89     log("bup: error: %s" % e)
90     sys.exit(2)
91
92 if not hash:
93     log("bup: error: commit %s not found.\n" % commit.decode('ascii'))
94     sys.exit(2)
95
96 pL = git.PackIdxList(git.repo(b'objects/pack'))
97 if not pL.exists(hash):
98     log("bup: error: commit %s not found.\n" % commit.decode('ascii'))
99     sys.exit(2)
100
101 tag_file = git.repo(b'refs/tags/' + tag_name)
102 try:
103     tag = open(tag_file, 'wb')
104 except OSError as e:
105     log("bup: error: could not create tag '%s': %s" % (path_msg(tag_name), e))
106     sys.exit(3)
107 with tag as tag:
108     tag.write(hexlify(hash))
109     tag.write(b'\n')