1
# Copyright (C) 2006, 2007, 2009, 2010, 2011 Canonical Ltd
3
# This program is free software; you can redistribute it and/or modify
4
# it under the terms of the GNU General Public License as published by
5
# the Free Software Foundation; either version 2 of the License, or
6
# (at your option) any later version.
8
# This program is distributed in the hope that it will be useful,
9
# but WITHOUT ANY WARRANTY; without even the implied warranty of
10
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11
# GNU General Public License for more details.
13
# You should have received a copy of the GNU General Public License
14
# along with this program; if not, write to the Free Software
15
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
17
"""Command which looks for unsigned commits by the current user, and signs them.
20
from __future__ import absolute_import
26
revision as _mod_revision,
28
from .commands import Command
29
from .option import Option
30
from .i18n import gettext, ngettext
31
from .sixish import text_type
34
class cmd_sign_my_commits(Command):
35
__doc__ = """Sign all commits by a given committer.
37
If location is not specified the local tree is used.
38
If committer is not specified the default committer is used.
40
This does not sign commits that already have signatures.
42
# Note that this signs everything on the branch's ancestry
43
# (both mainline and merged), but not other revisions that may be in the
48
help='Don\'t actually sign anything, just print'
49
' the revisions that would be signed.'),
51
takes_args = ['location?', 'committer?']
53
def run(self, location=None, committer=None, dry_run=False):
55
bzrdir = controldir.ControlDir.open_containing('.')[0]
57
# Passed in locations should be exact
58
bzrdir = controldir.ControlDir.open(location)
59
branch = bzrdir.open_branch()
60
repo = branch.repository
61
branch_config = branch.get_config_stack()
64
committer = branch_config.get('email')
65
gpg_strategy = gpg.GPGStrategy(branch_config)
68
with repo.lock_write():
69
graph = repo.get_graph()
70
repo.start_write_group()
72
for rev_id, parents in graph.iter_ancestry(
73
[branch.last_revision()]):
74
if _mod_revision.is_null(rev_id):
79
if repo.has_signature_for_revision_id(rev_id):
81
rev = repo.get_revision(rev_id)
82
if rev.committer != committer:
84
# We have a revision without a signature who has a
85
# matching committer, start signing
86
self.outf.write("%s\n" % rev_id)
89
repo.sign_revision(rev_id, gpg_strategy)
91
repo.abort_write_group()
94
repo.commit_write_group()
96
ngettext('Signed %d revision.\n', 'Signed %d revisions.\n', count) %
100
class cmd_verify_signatures(Command):
101
__doc__ = """Verify all commit signatures.
103
Verifies that all commits in the branch are signed by known GnuPG keys.
107
Option('acceptable-keys',
108
help='Comma separated list of GPG key patterns which are'
109
' acceptable for verification.',
115
takes_args = ['location?']
117
def run(self, acceptable_keys=None, revision=None, verbose=None,
119
bzrdir = controldir.ControlDir.open_containing(location)[0]
120
branch = bzrdir.open_branch()
121
repo = branch.repository
122
branch_config = branch.get_config_stack()
123
gpg_strategy = gpg.GPGStrategy(branch_config)
125
gpg_strategy.set_acceptable_keys(acceptable_keys)
128
self.outf.write(string + "\n")
129
def write_verbose(string):
130
self.outf.write(" " + string + "\n")
132
self.add_cleanup(repo.lock_read().unlock)
133
#get our list of revisions
135
if revision is not None:
136
if len(revision) == 1:
137
revno, rev_id = revision[0].in_history(branch)
138
revisions.append(rev_id)
139
elif len(revision) == 2:
140
from_revno, from_revid = revision[0].in_history(branch)
141
to_revno, to_revid = revision[1].in_history(branch)
143
to_revno = branch.revno()
144
if from_revno is None or to_revno is None:
145
raise errors.BzrCommandError(gettext(
146
'Cannot verify a range of non-revision-history revisions'))
147
for revno in range(from_revno, to_revno + 1):
148
revisions.append(branch.get_rev_id(revno))
150
#all revisions by default including merges
151
graph = repo.get_graph()
153
for rev_id, parents in graph.iter_ancestry(
154
[branch.last_revision()]):
155
if _mod_revision.is_null(rev_id):
160
revisions.append(rev_id)
161
count, result, all_verifiable = gpg.bulk_verify_signatures(
162
repo, revisions, gpg_strategy)
164
write(gettext("All commits signed with verifiable keys"))
166
for message in gpg.verbose_valid_message(result):
167
write_verbose(message)
170
write(gpg.valid_commits_message(count))
172
for message in gpg.verbose_valid_message(result):
173
write_verbose(message)
174
write(gpg.expired_commit_message(count))
176
for message in gpg.verbose_expired_key_message(result, repo):
177
write_verbose(message)
178
write(gpg.unknown_key_message(count))
180
for message in gpg.verbose_missing_key_message(result):
181
write_verbose(message)
182
write(gpg.commit_not_valid_message(count))
184
for message in gpg.verbose_not_valid_message(result, repo):
185
write_verbose(message)
186
write(gpg.commit_not_signed_message(count))
188
for message in gpg.verbose_not_signed_message(result, repo):
189
write_verbose(message)