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 that signs unsigned commits by the current user. """
19
from __future__ import absolute_import
25
repository as _mod_repository,
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
with _mod_repository.WriteGroup(repo):
71
for rev_id, parents in graph.iter_ancestry(
72
[branch.last_revision()]):
73
if _mod_revision.is_null(rev_id):
78
if repo.has_signature_for_revision_id(rev_id):
80
rev = repo.get_revision(rev_id)
81
if rev.committer != committer:
83
# We have a revision without a signature who has a
84
# matching committer, start signing
85
self.outf.write("%s\n" % rev_id)
88
repo.sign_revision(rev_id, gpg_strategy)
90
ngettext('Signed %d revision.\n', 'Signed %d revisions.\n',
94
class cmd_verify_signatures(Command):
95
__doc__ = """Verify all commit signatures.
97
Verifies that all commits in the branch are signed by known GnuPG keys.
101
Option('acceptable-keys',
102
help='Comma separated list of GPG key patterns which are'
103
' acceptable for verification.',
109
takes_args = ['location?']
111
def run(self, acceptable_keys=None, revision=None, verbose=None,
113
bzrdir = controldir.ControlDir.open_containing(location)[0]
114
branch = bzrdir.open_branch()
115
repo = branch.repository
116
branch_config = branch.get_config_stack()
117
gpg_strategy = gpg.GPGStrategy(branch_config)
119
gpg_strategy.set_acceptable_keys(acceptable_keys)
122
self.outf.write(string + "\n")
124
def write_verbose(string):
125
self.outf.write(" " + string + "\n")
127
self.add_cleanup(repo.lock_read().unlock)
128
# get our list of revisions
130
if revision is not None:
131
if len(revision) == 1:
132
revno, rev_id = revision[0].in_history(branch)
133
revisions.append(rev_id)
134
elif len(revision) == 2:
135
from_revno, from_revid = revision[0].in_history(branch)
136
to_revno, to_revid = revision[1].in_history(branch)
138
to_revno = branch.revno()
139
if from_revno is None or to_revno is None:
140
raise errors.BzrCommandError(
141
gettext('Cannot verify a range of non-revision-history'
143
for revno in range(from_revno, to_revno + 1):
144
revisions.append(branch.get_rev_id(revno))
146
# all revisions by default including merges
147
graph = repo.get_graph()
149
for rev_id, parents in graph.iter_ancestry(
150
[branch.last_revision()]):
151
if _mod_revision.is_null(rev_id):
156
revisions.append(rev_id)
157
count, result, all_verifiable = gpg.bulk_verify_signatures(
158
repo, revisions, gpg_strategy)
160
write(gettext("All commits signed with verifiable keys"))
162
for message in gpg.verbose_valid_message(result):
163
write_verbose(message)
166
write(gpg.valid_commits_message(count))
168
for message in gpg.verbose_valid_message(result):
169
write_verbose(message)
170
write(gpg.expired_commit_message(count))
172
for message in gpg.verbose_expired_key_message(result, repo):
173
write_verbose(message)
174
write(gpg.unknown_key_message(count))
176
for message in gpg.verbose_missing_key_message(result):
177
write_verbose(message)
178
write(gpg.commit_not_valid_message(count))
180
for message in gpg.verbose_not_valid_message(result, repo):
181
write_verbose(message)
182
write(gpg.commit_not_signed_message(count))
184
for message in gpg.verbose_not_signed_message(result, repo):
185
write_verbose(message)