mirror of
git://git.gnupg.org/gnupg.git
synced 2025-01-20 14:37:08 +01:00
824 lines
21 KiB
C
824 lines
21 KiB
C
/* keyedit.c - keyedit stuff
|
|
* Copyright (C) 1998 Free Software Foundation, Inc.
|
|
*
|
|
* This file is part of GNUPG.
|
|
*
|
|
* GNUPG is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* GNUPG is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA
|
|
*/
|
|
|
|
#include <config.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <errno.h>
|
|
#include <assert.h>
|
|
|
|
#include "options.h"
|
|
#include "packet.h"
|
|
#include "errors.h"
|
|
#include "iobuf.h"
|
|
#include "keydb.h"
|
|
#include "memory.h"
|
|
#include "util.h"
|
|
#include "main.h"
|
|
#include "filter.h"
|
|
#include "ttyio.h"
|
|
#include "i18n.h"
|
|
|
|
|
|
|
|
static void
|
|
show_fingerprint( PKT_public_key *pk )
|
|
{
|
|
byte *array, *p;
|
|
size_t i, n;
|
|
|
|
p = array = fingerprint_from_pk( pk, &n );
|
|
tty_printf(" Fingerprint:");
|
|
if( n == 20 ) {
|
|
for(i=0; i < n ; i++, i++, p += 2 ) {
|
|
if( i == 10 )
|
|
tty_printf(" ");
|
|
tty_printf(" %02X%02X", *p, p[1] );
|
|
}
|
|
}
|
|
else {
|
|
for(i=0; i < n ; i++, p++ ) {
|
|
if( i && !(i%8) )
|
|
tty_printf(" ");
|
|
tty_printf(" %02X", *p );
|
|
}
|
|
}
|
|
tty_printf("\n");
|
|
m_free(array);
|
|
}
|
|
|
|
|
|
/****************
|
|
* Ask whether the user is willing to sign the key. Return true if so.
|
|
*/
|
|
static int
|
|
sign_it_p( PKT_public_key *pk, PKT_user_id *uid )
|
|
{
|
|
char *answer;
|
|
int yes;
|
|
|
|
tty_printf("\n");
|
|
tty_printf(_("Are you really sure that you want to sign this key:\n\n"));
|
|
tty_printf("pub %4u%c/%08lX %s ",
|
|
nbits_from_pk( pk ),
|
|
pubkey_letter( pk->pubkey_algo ),
|
|
(ulong)keyid_from_pk( pk, NULL ),
|
|
datestr_from_pk( pk ) );
|
|
tty_print_string( uid->name, uid->len );
|
|
tty_printf("\n");
|
|
show_fingerprint(pk);
|
|
tty_printf("\n");
|
|
answer = tty_get(_("Sign this key? "));
|
|
tty_kill_prompt();
|
|
yes = answer_is_yes(answer);
|
|
m_free(answer);
|
|
return yes;
|
|
}
|
|
|
|
|
|
/****************
|
|
* Check the keysigs and set the flags to indicate errors.
|
|
* Usage of nodes flag bits:
|
|
* Bit 0 = bad signature
|
|
* 1 = no public key
|
|
* 2 = other error
|
|
* Returns true if error found.
|
|
*/
|
|
static int
|
|
check_all_keysigs( KBNODE keyblock )
|
|
{
|
|
KBNODE kbctx;
|
|
KBNODE node;
|
|
int rc;
|
|
int inv_sigs = 0;
|
|
int no_key = 0;
|
|
int oth_err = 0;
|
|
|
|
for( kbctx=NULL; (node=walk_kbnode( keyblock, &kbctx, 0)) ; ) {
|
|
if( node->pkt->pkttype == PKT_SIGNATURE
|
|
&& (node->pkt->pkt.signature->sig_class&~3) == 0x10 ) {
|
|
PKT_signature *sig = node->pkt->pkt.signature;
|
|
int sigrc;
|
|
|
|
tty_printf("sig");
|
|
switch( (rc = check_key_signature( keyblock, node,NULL)) ) {
|
|
case 0: node->flag = 0; sigrc = '!'; break;
|
|
case G10ERR_BAD_SIGN: inv_sigs++; node->flag = 1; sigrc = '-'; break;
|
|
case G10ERR_NO_PUBKEY: no_key++; node->flag = 2; sigrc = '?'; break;
|
|
default: oth_err++; node->flag = 4; sigrc = '%'; break;
|
|
}
|
|
tty_printf("%c %08lX %s ",
|
|
sigrc, sig->keyid[1], datestr_from_sig(sig));
|
|
if( sigrc == '%' )
|
|
tty_printf("[%s] ", g10_errstr(rc) );
|
|
else if( sigrc == '?' )
|
|
;
|
|
else {
|
|
size_t n;
|
|
char *p = get_user_id( sig->keyid, &n );
|
|
tty_print_string( p, n > 40? 40 : n );
|
|
m_free(p);
|
|
}
|
|
tty_printf("\n");
|
|
/* FIXME: update the trustdb */
|
|
}
|
|
}
|
|
if( inv_sigs )
|
|
tty_printf(_("%d bad signatures\n"), inv_sigs );
|
|
if( no_key )
|
|
tty_printf(_("No public key for %d signatures\n"), no_key );
|
|
if( oth_err )
|
|
tty_printf(_("%d signatures not checked due to errors\n"), oth_err );
|
|
return inv_sigs || no_key || oth_err;
|
|
}
|
|
|
|
|
|
/****************
|
|
* Ask and remove invalid signatures that are to be removed.
|
|
*/
|
|
static int
|
|
remove_keysigs( KBNODE keyblock, u32 *keyid, int all )
|
|
{
|
|
KBNODE kbctx;
|
|
KBNODE node;
|
|
char *answer;
|
|
int yes;
|
|
int count;
|
|
|
|
count = 0;
|
|
for( kbctx=NULL; (node=walk_kbnode( keyblock, &kbctx, 0)) ; ) {
|
|
if( ((node->flag & 7) || all )
|
|
&& node->pkt->pkttype == PKT_SIGNATURE
|
|
&& (node->pkt->pkt.signature->sig_class&~3) == 0x10 ) {
|
|
PKT_signature *sig = node->pkt->pkt.signature;
|
|
|
|
tty_printf("\n \"%08lX %s ",
|
|
sig->keyid[1], datestr_from_sig(sig));
|
|
if( node->flag & 6 )
|
|
tty_printf(_("[User name not available] "));
|
|
else {
|
|
size_t n;
|
|
char *p = get_user_id( sig->keyid, &n );
|
|
tty_print_string( p, n );
|
|
m_free(p);
|
|
}
|
|
tty_printf("\"\n");
|
|
if( node->flag & 1 )
|
|
tty_printf(_("This is a BAD signature!\n"));
|
|
else if( node->flag & 2 )
|
|
tty_printf(_("Public key not available.\n"));
|
|
else if( node->flag & 4 )
|
|
tty_printf(_("The signature could not be checked!\n"));
|
|
|
|
if( keyid[0] == sig->keyid[0] && keyid[1] == sig->keyid[1] ) {
|
|
tty_printf(_("Skipped self-signature\n"));
|
|
continue; /* do not remove self-signatures */
|
|
}
|
|
|
|
tty_printf("\n");
|
|
answer = tty_get(_("Remove this signature? "));
|
|
tty_kill_prompt();
|
|
if( answer_is_yes(answer) ) {
|
|
node->flag |= 128; /* use bit 7 to mark this node */
|
|
count++;
|
|
}
|
|
m_free(answer);
|
|
}
|
|
}
|
|
|
|
if( !count )
|
|
return 0; /* nothing to remove */
|
|
answer = tty_get(_("Do you really want to remove the selected signatures? "));
|
|
tty_kill_prompt();
|
|
yes = answer_is_yes(answer);
|
|
m_free(answer);
|
|
if( !yes )
|
|
return 0;
|
|
|
|
for( kbctx=NULL; (node=walk_kbnode( keyblock, &kbctx, 1)) ; ) {
|
|
if( node->flag & 128)
|
|
delete_kbnode(node );
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
|
|
/****************
|
|
* This function signs the key of USERNAME with all users listed in
|
|
* LOCUSR. If LOCUSR is NULL the default secret certificate will
|
|
* be used. This works on all keyrings, so there is no armor or
|
|
* compress stuff here.
|
|
*/
|
|
int
|
|
sign_key( const char *username, STRLIST locusr )
|
|
{
|
|
md_filter_context_t mfx;
|
|
int rc = 0;
|
|
SK_LIST sk_list = NULL;
|
|
SK_LIST sk_rover = NULL;
|
|
KBNODE keyblock = NULL;
|
|
KBNODE kbctx, node;
|
|
KBPOS kbpos;
|
|
PKT_public_key *pk;
|
|
u32 pk_keyid[2];
|
|
char *answer;
|
|
|
|
memset( &mfx, 0, sizeof mfx);
|
|
|
|
/* search the userid */
|
|
rc = find_keyblock_byname( &kbpos, username );
|
|
if( rc ) {
|
|
log_error(_("%s: user not found\n"), username );
|
|
goto leave;
|
|
}
|
|
|
|
/* build a list of all signators */
|
|
rc=build_sk_list( locusr, &sk_list, 0, 1 );
|
|
if( rc )
|
|
goto leave;
|
|
|
|
|
|
/* read the keyblock */
|
|
rc = read_keyblock( &kbpos, &keyblock );
|
|
if( rc ) {
|
|
log_error("error reading the certificate: %s\n", g10_errstr(rc) );
|
|
goto leave;
|
|
}
|
|
|
|
/* get the keyid from the keyblock */
|
|
node = find_kbnode( keyblock, PKT_PUBLIC_KEY );
|
|
if( !node ) {
|
|
log_error("Oops; public key not found anymore!\n");
|
|
rc = G10ERR_GENERAL;
|
|
goto leave;
|
|
}
|
|
|
|
pk = node->pkt->pkt.public_key;
|
|
keyid_from_pk( pk, pk_keyid );
|
|
tty_printf(_("Checking signatures of this public key certificate:\n"));
|
|
tty_printf("pub %4u%c/%08lX %s ",
|
|
nbits_from_pk( pk ),
|
|
pubkey_letter( pk->pubkey_algo ),
|
|
pk_keyid[1], datestr_from_pk(pk) );
|
|
{
|
|
size_t n;
|
|
char *p = get_user_id( pk_keyid, &n );
|
|
tty_print_string( p, n > 40? 40 : n );
|
|
m_free(p);
|
|
tty_printf("\n");
|
|
}
|
|
|
|
clear_kbnode_flags( keyblock );
|
|
if( check_all_keysigs( keyblock ) ) {
|
|
if( !opt.batch ) {
|
|
/* ask whether we really should do anything */
|
|
answer = tty_get(
|
|
_("To you want to remove some of the invalid sigs? "));
|
|
tty_kill_prompt();
|
|
if( answer_is_yes(answer) )
|
|
remove_keysigs( keyblock, pk_keyid, 0 );
|
|
m_free(answer);
|
|
}
|
|
}
|
|
|
|
/* check whether we it is possible to sign this key */
|
|
for( sk_rover = sk_list; sk_rover; sk_rover = sk_rover->next ) {
|
|
u32 akeyid[2];
|
|
|
|
keyid_from_sk( sk_rover->sk, akeyid );
|
|
for( kbctx=NULL; (node=walk_kbnode( keyblock, &kbctx, 0)) ; ) {
|
|
if( node->pkt->pkttype == PKT_USER_ID )
|
|
sk_rover->mark = 1;
|
|
else if( node->pkt->pkttype == PKT_SIGNATURE
|
|
&& (node->pkt->pkt.signature->sig_class&~3) == 0x10 ) {
|
|
if( akeyid[0] == node->pkt->pkt.signature->keyid[0]
|
|
&& akeyid[1] == node->pkt->pkt.signature->keyid[1] ) {
|
|
log_info(_("Already signed by keyid %08lX\n"),
|
|
(ulong)akeyid[1] );
|
|
sk_rover->mark = 0;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
for( sk_rover = sk_list; sk_rover; sk_rover = sk_rover->next ) {
|
|
if( sk_rover->mark )
|
|
break;
|
|
}
|
|
if( !sk_rover ) {
|
|
log_info(_("Nothing to sign\n"));
|
|
goto leave;
|
|
}
|
|
|
|
/* Loop over all signers and all user ids and sign */
|
|
/* FIXME: we have to change it: Present all user-ids and
|
|
* then ask whether all those ids shall be signed if the user
|
|
* answers yes, go and make a 0x14 sign class packet and remove
|
|
* old one-user-id-only-sigs (user should be noted of this
|
|
* condition while presenting the user-ids); if he had answered
|
|
* no, present each user-id in turn and ask which one should be signed
|
|
* (only one) - if there is already a single-user-sig, do nothing.
|
|
* (this is propably already out in the world) */
|
|
for( sk_rover = sk_list; sk_rover; sk_rover = sk_rover->next ) {
|
|
if( !sk_rover->mark )
|
|
continue;
|
|
for( kbctx=NULL; (node=walk_kbnode( keyblock, &kbctx, 0)) ; ) {
|
|
if( node->pkt->pkttype == PKT_USER_ID ) {
|
|
if( sign_it_p( pk, node->pkt->pkt.user_id ) ) {
|
|
PACKET *pkt;
|
|
PKT_signature *sig;
|
|
|
|
rc = make_keysig_packet( &sig, pk,
|
|
node->pkt->pkt.user_id,
|
|
NULL,
|
|
sk_rover->sk,
|
|
0x10, 0, NULL, NULL );
|
|
if( rc ) {
|
|
log_error("make_keysig_packet failed: %s\n", g10_errstr(rc));
|
|
goto leave;
|
|
}
|
|
|
|
pkt = m_alloc_clear( sizeof *pkt );
|
|
pkt->pkttype = PKT_SIGNATURE;
|
|
pkt->pkt.signature = sig;
|
|
insert_kbnode( node, new_kbnode(pkt), PKT_USER_ID );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
rc = update_keyblock( &kbpos, keyblock );
|
|
if( rc ) {
|
|
log_error("update_keyblock failed: %s\n", g10_errstr(rc) );
|
|
goto leave;
|
|
}
|
|
|
|
leave:
|
|
release_kbnode( keyblock );
|
|
release_sk_list( sk_list );
|
|
md_close( mfx.md );
|
|
return rc;
|
|
}
|
|
|
|
|
|
|
|
int
|
|
edit_keysigs( const char *username )
|
|
{
|
|
int rc = 0;
|
|
KBNODE keyblock = NULL;
|
|
KBNODE node;
|
|
KBPOS kbpos;
|
|
PKT_public_key *pk;
|
|
u32 pk_keyid[2];
|
|
|
|
/* search the userid */
|
|
rc = find_keyblock_byname( &kbpos, username );
|
|
if( rc ) {
|
|
log_error(_("%s: user not found\n"), username );
|
|
goto leave;
|
|
}
|
|
|
|
/* read the keyblock */
|
|
rc = read_keyblock( &kbpos, &keyblock );
|
|
if( rc ) {
|
|
log_error("%s: certificate read problem: %s\n", username, g10_errstr(rc) );
|
|
goto leave;
|
|
}
|
|
|
|
/* get the keyid from the keyblock */
|
|
node = find_kbnode( keyblock, PKT_PUBLIC_KEY );
|
|
if( !node ) {
|
|
log_error("Oops; public key not found anymore!\n");
|
|
rc = G10ERR_GENERAL;
|
|
goto leave;
|
|
}
|
|
|
|
pk = node->pkt->pkt.public_key;
|
|
keyid_from_pk( pk, pk_keyid );
|
|
tty_printf(_("Checking signatures of this public key certificate:\n"));
|
|
tty_printf("pub %4u%c/%08lX %s ",
|
|
nbits_from_pk( pk ),
|
|
pubkey_letter( pk->pubkey_algo ),
|
|
pk_keyid[1], datestr_from_pk(pk) );
|
|
{
|
|
size_t n;
|
|
char *p = get_user_id( pk_keyid, &n );
|
|
tty_print_string( p, n > 40? 40 : n );
|
|
m_free(p);
|
|
tty_printf("\n");
|
|
}
|
|
|
|
clear_kbnode_flags( keyblock );
|
|
check_all_keysigs( keyblock );
|
|
if( remove_keysigs( keyblock, pk_keyid, 1 ) ) {
|
|
rc = update_keyblock( &kbpos, keyblock );
|
|
if( rc ) {
|
|
log_error("update_keyblock failed: %s\n", g10_errstr(rc) );
|
|
goto leave;
|
|
}
|
|
}
|
|
|
|
leave:
|
|
release_kbnode( keyblock );
|
|
return rc;
|
|
}
|
|
|
|
|
|
/****************
|
|
* Delete a public or secret key from a keyring.
|
|
*/
|
|
int
|
|
delete_key( const char *username, int secret )
|
|
{
|
|
int rc = 0;
|
|
KBNODE keyblock = NULL;
|
|
KBNODE node;
|
|
KBPOS kbpos;
|
|
PKT_public_key *pk = NULL;
|
|
PKT_secret_key *sk = NULL;
|
|
u32 keyid[2];
|
|
int okay=0;
|
|
|
|
/* search the userid */
|
|
rc = secret? find_secret_keyblock_byname( &kbpos, username )
|
|
: find_keyblock_byname( &kbpos, username );
|
|
if( rc ) {
|
|
log_error(_("%s: user not found\n"), username );
|
|
goto leave;
|
|
}
|
|
|
|
/* read the keyblock */
|
|
rc = read_keyblock( &kbpos, &keyblock );
|
|
if( rc ) {
|
|
log_error("%s: read problem: %s\n", username, g10_errstr(rc) );
|
|
goto leave;
|
|
}
|
|
|
|
/* get the keyid from the keyblock */
|
|
node = find_kbnode( keyblock, secret? PKT_SECRET_KEY:PKT_PUBLIC_KEY );
|
|
if( !node ) {
|
|
log_error("Oops; key not found anymore!\n");
|
|
rc = G10ERR_GENERAL;
|
|
goto leave;
|
|
}
|
|
|
|
if( secret ) {
|
|
sk = node->pkt->pkt.secret_key;
|
|
keyid_from_sk( sk, keyid );
|
|
}
|
|
else {
|
|
pk = node->pkt->pkt.public_key;
|
|
keyid_from_pk( pk, keyid );
|
|
rc = seckey_available( keyid );
|
|
if( !rc ) {
|
|
log_error(_(
|
|
"there is a secret key for this public key!\n"));
|
|
log_info(_(
|
|
"use option \"--delete-secret-key\" to delete it first.\n"));
|
|
rc = -1;
|
|
}
|
|
else if( rc != G10ERR_NO_SECKEY )
|
|
log_error("%s: get secret key: %s\n", username, g10_errstr(rc) );
|
|
else
|
|
rc = 0;
|
|
}
|
|
|
|
if( rc )
|
|
rc = 0;
|
|
else if( opt.batch && secret )
|
|
log_error(_("can't do that in batch-mode\n"));
|
|
else if( opt.batch && opt.answer_yes )
|
|
okay++;
|
|
else if( opt.batch )
|
|
log_error(_("can't do that in batch-mode without \"--yes\"\n"));
|
|
else {
|
|
char *p;
|
|
size_t n;
|
|
|
|
if( secret )
|
|
tty_printf("sec %4u%c/%08lX %s ",
|
|
nbits_from_sk( sk ),
|
|
pubkey_letter( sk->pubkey_algo ),
|
|
keyid[1], datestr_from_sk(sk) );
|
|
else
|
|
tty_printf("pub %4u%c/%08lX %s ",
|
|
nbits_from_pk( pk ),
|
|
pubkey_letter( pk->pubkey_algo ),
|
|
keyid[1], datestr_from_pk(pk) );
|
|
p = get_user_id( keyid, &n );
|
|
tty_print_string( p, n );
|
|
m_free(p);
|
|
tty_printf("\n\n");
|
|
|
|
p = tty_get(_("Delete this key from the keyring? "));
|
|
tty_kill_prompt();
|
|
if( secret && answer_is_yes(p)) {
|
|
/* I think it is not required to check a passphrase; if
|
|
* the user is so stupid as to let others access his secret keyring
|
|
* (and has no backup) - it is up him to read some very
|
|
* basic texts about security.
|
|
*/
|
|
m_free(p);
|
|
p = tty_get(_("This is a secret key! - really delete? "));
|
|
}
|
|
if( answer_is_yes(p) )
|
|
okay++;
|
|
m_free(p);
|
|
}
|
|
|
|
|
|
if( okay ) {
|
|
rc = delete_keyblock( &kbpos );
|
|
if( rc ) {
|
|
log_error("delete_keyblock failed: %s\n", g10_errstr(rc) );
|
|
goto leave;
|
|
}
|
|
}
|
|
|
|
leave:
|
|
release_kbnode( keyblock );
|
|
return rc;
|
|
}
|
|
|
|
|
|
int
|
|
change_passphrase( const char *username )
|
|
{
|
|
int rc = 0;
|
|
KBNODE keyblock = NULL;
|
|
KBNODE node;
|
|
KBPOS kbpos;
|
|
PKT_secret_key *sk;
|
|
u32 keyid[2];
|
|
char *answer;
|
|
int changed=0;
|
|
char *passphrase = NULL;
|
|
|
|
/* find the userid */
|
|
rc = find_secret_keyblock_byname( &kbpos, username );
|
|
if( rc ) {
|
|
log_error("secret key for user '%s' not found\n", username );
|
|
goto leave;
|
|
}
|
|
|
|
/* read the keyblock */
|
|
rc = read_keyblock( &kbpos, &keyblock );
|
|
if( rc ) {
|
|
log_error("error reading the certificate: %s\n", g10_errstr(rc) );
|
|
goto leave;
|
|
}
|
|
|
|
/* get the keyid from the keyblock */
|
|
node = find_kbnode( keyblock, PKT_SECRET_KEY );
|
|
if( !node ) {
|
|
log_error("Oops; secret key not found anymore!\n");
|
|
rc = G10ERR_GENERAL;
|
|
goto leave;
|
|
}
|
|
|
|
sk = node->pkt->pkt.secret_key;
|
|
keyid_from_sk( sk, keyid );
|
|
tty_printf("sec %4u%c/%08lX %s ",
|
|
nbits_from_sk( sk ),
|
|
pubkey_letter( sk->pubkey_algo ),
|
|
keyid[1], datestr_from_sk(sk) );
|
|
{
|
|
size_t n;
|
|
char *p = get_user_id( keyid, &n );
|
|
tty_print_string( p, n );
|
|
m_free(p);
|
|
tty_printf("\n");
|
|
}
|
|
for(node=keyblock; node; node = node->next ) {
|
|
if( node->pkt->pkttype == PKT_SECRET_SUBKEY ) {
|
|
PKT_secret_key *subsk = node->pkt->pkt.secret_key;
|
|
keyid_from_sk( subsk, keyid );
|
|
tty_printf("sub %4u%c/%08lX %s\n",
|
|
nbits_from_sk( subsk ),
|
|
pubkey_letter( subsk->pubkey_algo ),
|
|
keyid[1], datestr_from_sk(subsk) );
|
|
}
|
|
}
|
|
|
|
clear_kbnode_flags( keyblock );
|
|
switch( is_secret_key_protected( sk ) ) {
|
|
case -1:
|
|
rc = G10ERR_PUBKEY_ALGO;
|
|
break;
|
|
case 0:
|
|
tty_printf(_("This key is not protected.\n"));
|
|
break;
|
|
default:
|
|
tty_printf(_("Key is protected.\n"));
|
|
rc = check_secret_key( sk );
|
|
if( !rc )
|
|
passphrase = get_last_passphrase();
|
|
break;
|
|
}
|
|
|
|
/* unprotect all subkeys (use the supplied passphrase or ask)*/
|
|
for(node=keyblock; node; node = node->next ) {
|
|
if( node->pkt->pkttype == PKT_SECRET_SUBKEY ) {
|
|
PKT_secret_key *subsk = node->pkt->pkt.secret_key;
|
|
set_next_passphrase( passphrase );
|
|
rc = check_secret_key( subsk );
|
|
if( rc )
|
|
break;
|
|
}
|
|
}
|
|
|
|
if( rc )
|
|
tty_printf(_("Can't edit this key: %s\n"), g10_errstr(rc));
|
|
else {
|
|
DEK *dek = NULL;
|
|
STRING2KEY *s2k = m_alloc_secure( sizeof *s2k );
|
|
|
|
tty_printf(_("Enter the new passphrase for this secret key.\n\n") );
|
|
|
|
set_next_passphrase( NULL );
|
|
for(;;) {
|
|
s2k->mode = 1;
|
|
s2k->hash_algo = DIGEST_ALGO_RMD160;
|
|
dek = passphrase_to_dek( NULL, CIPHER_ALGO_BLOWFISH, s2k, 2 );
|
|
if( !dek ) {
|
|
tty_printf(_("passphrase not correctly repeated; try again.\n"));
|
|
}
|
|
else if( !dek->keylen ) {
|
|
rc = 0;
|
|
tty_printf(_( "You don't want a passphrase -"
|
|
" this is probably a *bad* idea!\n\n"));
|
|
answer = tty_get(_("Do you really want to do this? "));
|
|
tty_kill_prompt();
|
|
if( answer_is_yes(answer) )
|
|
changed++;
|
|
m_free(answer);
|
|
break;
|
|
}
|
|
else { /* okay */
|
|
sk->protect.algo = dek->algo;
|
|
sk->protect.s2k = *s2k;
|
|
rc = protect_secret_key( sk, dek );
|
|
for(node=keyblock; !rc && node; node = node->next ) {
|
|
if( node->pkt->pkttype == PKT_SECRET_SUBKEY ) {
|
|
PKT_secret_key *subsk = node->pkt->pkt.secret_key;
|
|
subsk->protect.algo = dek->algo;
|
|
subsk->protect.s2k = *s2k;
|
|
rc = protect_secret_key( subsk, dek );
|
|
}
|
|
}
|
|
if( rc )
|
|
log_error("protect_secret_key failed: %s\n", g10_errstr(rc) );
|
|
else
|
|
changed++;
|
|
break;
|
|
}
|
|
}
|
|
m_free(s2k);
|
|
m_free(dek);
|
|
}
|
|
|
|
|
|
if( changed ) {
|
|
rc = update_keyblock( &kbpos, keyblock );
|
|
if( rc ) {
|
|
log_error("update_keyblock failed: %s\n", g10_errstr(rc) );
|
|
goto leave;
|
|
}
|
|
}
|
|
|
|
leave:
|
|
m_free( passphrase );
|
|
release_kbnode( keyblock );
|
|
set_next_passphrase( NULL );
|
|
return rc;
|
|
}
|
|
|
|
|
|
/****************
|
|
* Create a signature packet for the given public key certificate
|
|
* and the user id and return it in ret_sig. User signature class SIGCLASS
|
|
* user-id is not used (and may be NULL if sigclass is 0x20)
|
|
* If digest_algo is 0 the function selects an appropriate one.
|
|
*/
|
|
int
|
|
make_keysig_packet( PKT_signature **ret_sig, PKT_public_key *pk,
|
|
PKT_user_id *uid, PKT_public_key *subpk,
|
|
PKT_secret_key *sk,
|
|
int sigclass, int digest_algo,
|
|
int (*mksubpkt)(PKT_signature *, void *), void *opaque
|
|
)
|
|
{
|
|
PKT_signature *sig;
|
|
int rc=0;
|
|
MD_HANDLE md;
|
|
|
|
assert( (sigclass >= 0x10 && sigclass <= 0x13)
|
|
|| sigclass == 0x20 || sigclass == 0x18 );
|
|
if( !digest_algo ) {
|
|
switch( sk->pubkey_algo ) {
|
|
case PUBKEY_ALGO_DSA: digest_algo = DIGEST_ALGO_SHA1; break;
|
|
case PUBKEY_ALGO_RSA_S:
|
|
case PUBKEY_ALGO_RSA: digest_algo = DIGEST_ALGO_MD5; break;
|
|
default: digest_algo = DIGEST_ALGO_RMD160; break;
|
|
}
|
|
}
|
|
md = md_open( digest_algo, 0 );
|
|
|
|
/* hash the public key certificate and the user id */
|
|
hash_public_key( md, pk );
|
|
if( sigclass == 0x18 ) { /* subkey binding */
|
|
hash_public_key( md, subpk );
|
|
}
|
|
else if( sigclass != 0x20 ) {
|
|
if( sk->version >=4 ) {
|
|
byte buf[5];
|
|
buf[0] = 0xb4; /* indicates a userid packet */
|
|
buf[1] = uid->len >> 24; /* always use 4 length bytes */
|
|
buf[2] = uid->len >> 16;
|
|
buf[3] = uid->len >> 8;
|
|
buf[4] = uid->len;
|
|
md_write( md, buf, 5 );
|
|
}
|
|
md_write( md, uid->name, uid->len );
|
|
}
|
|
/* and make the signature packet */
|
|
sig = m_alloc_clear( sizeof *sig );
|
|
sig->version = sk->version;
|
|
keyid_from_sk( sk, sig->keyid );
|
|
sig->pubkey_algo = sk->pubkey_algo;
|
|
sig->digest_algo = digest_algo;
|
|
sig->timestamp = make_timestamp();
|
|
sig->sig_class = sigclass;
|
|
if( sig->version >= 4 )
|
|
build_sig_subpkt_from_sig( sig );
|
|
|
|
if( sig->version >= 4 && mksubpkt )
|
|
rc = (*mksubpkt)( sig, opaque );
|
|
|
|
if( !rc ) {
|
|
if( sig->version >= 4 )
|
|
md_putc( md, sig->version );
|
|
md_putc( md, sig->sig_class );
|
|
if( sig->version < 4 ) {
|
|
u32 a = sig->timestamp;
|
|
md_putc( md, (a >> 24) & 0xff );
|
|
md_putc( md, (a >> 16) & 0xff );
|
|
md_putc( md, (a >> 8) & 0xff );
|
|
md_putc( md, a & 0xff );
|
|
}
|
|
else {
|
|
byte buf[6];
|
|
size_t n;
|
|
|
|
md_putc( md, sig->pubkey_algo );
|
|
md_putc( md, sig->digest_algo );
|
|
if( sig->hashed_data ) {
|
|
n = (sig->hashed_data[0] << 8) | sig->hashed_data[1];
|
|
md_write( md, sig->hashed_data, n+2 );
|
|
n += 6;
|
|
}
|
|
else
|
|
n = 6;
|
|
/* add some magic */
|
|
buf[0] = sig->version;
|
|
buf[1] = 0xff;
|
|
buf[2] = n >> 24; /* hmmm, n is only 16 bit, so this is always 0 */
|
|
buf[3] = n >> 16;
|
|
buf[4] = n >> 8;
|
|
buf[5] = n;
|
|
md_write( md, buf, 6 );
|
|
|
|
}
|
|
md_final(md);
|
|
|
|
rc = complete_sig( sig, sk, md );
|
|
}
|
|
|
|
md_close( md );
|
|
if( rc )
|
|
free_seckey_enc( sig );
|
|
else
|
|
*ret_sig = sig;
|
|
return rc;
|
|
}
|
|
|