1
0
mirror of git://git.gnupg.org/gnupg.git synced 2025-01-26 15:37:03 +01:00
gnupg/g10/keyedit.c

824 lines
21 KiB
C
Raw Normal View History

1998-04-20 15:52:20 +00:00
/* 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
1998-06-29 12:30:57 +00:00
show_fingerprint( PKT_public_key *pk )
1998-04-20 15:52:20 +00:00
{
byte *array, *p;
size_t i, n;
1998-06-29 12:30:57 +00:00
p = array = fingerprint_from_pk( pk, &n );
1998-04-20 15:52:20 +00:00
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
1998-06-29 12:30:57 +00:00
sign_it_p( PKT_public_key *pk, PKT_user_id *uid )
1998-04-20 15:52:20 +00:00
{
char *answer;
int yes;
1998-07-06 10:23:57 +00:00
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 ",
1998-06-29 12:30:57 +00:00
nbits_from_pk( pk ),
pubkey_letter( pk->pubkey_algo ),
(ulong)keyid_from_pk( pk, NULL ),
datestr_from_pk( pk ) );
1998-04-20 15:52:20 +00:00
tty_print_string( uid->name, uid->len );
tty_printf("\n");
1998-06-29 12:30:57 +00:00
show_fingerprint(pk);
1998-04-20 15:52:20 +00:00
tty_printf("\n");
1998-07-06 10:23:57 +00:00
answer = tty_get(_("Sign this key? "));
1998-04-20 15:52:20 +00:00
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 )
1998-07-06 10:23:57 +00:00
tty_printf(_("%d bad signatures\n"), inv_sigs );
1998-04-20 15:52:20 +00:00
if( no_key )
1998-07-06 10:23:57 +00:00
tty_printf(_("No public key for %d signatures\n"), no_key );
1998-04-20 15:52:20 +00:00
if( oth_err )
1998-07-06 10:23:57 +00:00
tty_printf(_("%d signatures not checked due to errors\n"), oth_err );
1998-04-20 15:52:20 +00:00
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 )
1998-07-06 10:23:57 +00:00
tty_printf(_("[User name not available] "));
1998-04-20 15:52:20 +00:00
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 )
1998-07-06 10:23:57 +00:00
tty_printf(_("This is a BAD signature!\n"));
1998-04-20 15:52:20 +00:00
else if( node->flag & 2 )
1998-07-06 10:23:57 +00:00
tty_printf(_("Public key not available.\n"));
1998-04-20 15:52:20 +00:00
else if( node->flag & 4 )
1998-07-06 10:23:57 +00:00
tty_printf(_("The signature could not be checked!\n"));
1998-04-20 15:52:20 +00:00
1998-05-13 17:53:36 +00:00
if( keyid[0] == sig->keyid[0] && keyid[1] == sig->keyid[1] ) {
1998-07-06 10:23:57 +00:00
tty_printf(_("Skipped self-signature\n"));
1998-04-20 15:52:20 +00:00
continue; /* do not remove self-signatures */
1998-05-13 17:53:36 +00:00
}
1998-04-20 15:52:20 +00:00
1998-07-06 10:23:57 +00:00
tty_printf("\n");
answer = tty_get(_("Remove this signature? "));
1998-04-20 15:52:20 +00:00
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 */
1998-07-06 10:23:57 +00:00
answer = tty_get(_("Do you really want to remove the selected signatures? "));
1998-04-20 15:52:20 +00:00
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;
1998-06-29 12:30:57 +00:00
SK_LIST sk_list = NULL;
SK_LIST sk_rover = NULL;
1998-04-20 15:52:20 +00:00
KBNODE keyblock = NULL;
KBNODE kbctx, node;
KBPOS kbpos;
1998-06-29 12:30:57 +00:00
PKT_public_key *pk;
u32 pk_keyid[2];
1998-04-20 15:52:20 +00:00
char *answer;
memset( &mfx, 0, sizeof mfx);
/* search the userid */
rc = find_keyblock_byname( &kbpos, username );
if( rc ) {
1998-07-06 10:23:57 +00:00
log_error(_("%s: user not found\n"), username );
1998-04-20 15:52:20 +00:00
goto leave;
}
/* build a list of all signators */
1998-06-29 12:30:57 +00:00
rc=build_sk_list( locusr, &sk_list, 0, 1 );
1998-04-20 15:52:20 +00:00
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 */
1998-06-29 12:30:57 +00:00
node = find_kbnode( keyblock, PKT_PUBLIC_KEY );
1998-04-20 15:52:20 +00:00
if( !node ) {
log_error("Oops; public key not found anymore!\n");
rc = G10ERR_GENERAL;
goto leave;
}
1998-06-29 12:30:57 +00:00
pk = node->pkt->pkt.public_key;
keyid_from_pk( pk, pk_keyid );
1998-07-06 10:23:57 +00:00
tty_printf(_("Checking signatures of this public key certificate:\n"));
1998-04-20 15:52:20 +00:00
tty_printf("pub %4u%c/%08lX %s ",
1998-06-29 12:30:57 +00:00
nbits_from_pk( pk ),
pubkey_letter( pk->pubkey_algo ),
pk_keyid[1], datestr_from_pk(pk) );
1998-04-20 15:52:20 +00:00
{
size_t n;
1998-06-29 12:30:57 +00:00
char *p = get_user_id( pk_keyid, &n );
1998-04-20 15:52:20 +00:00
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 */
1998-07-06 10:23:57 +00:00
answer = tty_get(
_("To you want to remove some of the invalid sigs? "));
1998-04-20 15:52:20 +00:00
tty_kill_prompt();
if( answer_is_yes(answer) )
1998-06-29 12:30:57 +00:00
remove_keysigs( keyblock, pk_keyid, 0 );
1998-04-20 15:52:20 +00:00
m_free(answer);
}
}
1998-04-25 08:08:35 +00:00
/* check whether we it is possible to sign this key */
1998-06-29 12:30:57 +00:00
for( sk_rover = sk_list; sk_rover; sk_rover = sk_rover->next ) {
1998-04-20 15:52:20 +00:00
u32 akeyid[2];
1998-06-29 12:30:57 +00:00
keyid_from_sk( sk_rover->sk, akeyid );
1998-04-20 15:52:20 +00:00
for( kbctx=NULL; (node=walk_kbnode( keyblock, &kbctx, 0)) ; ) {
1998-04-25 08:08:35 +00:00
if( node->pkt->pkttype == PKT_USER_ID )
1998-06-29 12:30:57 +00:00
sk_rover->mark = 1;
1998-04-25 08:08:35 +00:00
else if( node->pkt->pkttype == PKT_SIGNATURE
1998-04-20 15:52:20 +00:00
&& (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] ) {
1998-07-06 10:23:57 +00:00
log_info(_("Already signed by keyid %08lX\n"),
1998-04-20 15:52:20 +00:00
(ulong)akeyid[1] );
1998-06-29 12:30:57 +00:00
sk_rover->mark = 0;
1998-04-20 15:52:20 +00:00
}
}
}
}
1998-06-29 12:30:57 +00:00
for( sk_rover = sk_list; sk_rover; sk_rover = sk_rover->next ) {
if( sk_rover->mark )
1998-04-20 15:52:20 +00:00
break;
}
1998-06-29 12:30:57 +00:00
if( !sk_rover ) {
1998-07-06 10:23:57 +00:00
log_info(_("Nothing to sign\n"));
1998-04-20 15:52:20 +00:00
goto leave;
}
/* Loop over all signers and all user ids and sign */
1998-04-25 08:08:35 +00:00
/* 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) */
1998-06-29 12:30:57 +00:00
for( sk_rover = sk_list; sk_rover; sk_rover = sk_rover->next ) {
if( !sk_rover->mark )
1998-04-20 15:52:20 +00:00
continue;
for( kbctx=NULL; (node=walk_kbnode( keyblock, &kbctx, 0)) ; ) {
if( node->pkt->pkttype == PKT_USER_ID ) {
1998-06-29 12:30:57 +00:00
if( sign_it_p( pk, node->pkt->pkt.user_id ) ) {
1998-04-20 15:52:20 +00:00
PACKET *pkt;
PKT_signature *sig;
1998-06-29 12:30:57 +00:00
rc = make_keysig_packet( &sig, pk,
1998-04-20 15:52:20 +00:00
node->pkt->pkt.user_id,
1998-05-26 13:38:00 +00:00
NULL,
1998-06-29 12:30:57 +00:00
sk_rover->sk,
1998-06-25 10:19:08 +00:00
0x10, 0, NULL, NULL );
1998-04-20 15:52:20 +00:00
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 );
1998-06-29 12:30:57 +00:00
release_sk_list( sk_list );
1998-04-20 15:52:20 +00:00
md_close( mfx.md );
return rc;
}
int
edit_keysigs( const char *username )
{
int rc = 0;
KBNODE keyblock = NULL;
KBNODE node;
KBPOS kbpos;
1998-06-29 12:30:57 +00:00
PKT_public_key *pk;
u32 pk_keyid[2];
1998-04-20 15:52:20 +00:00
/* search the userid */
rc = find_keyblock_byname( &kbpos, username );
if( rc ) {
1998-07-06 10:23:57 +00:00
log_error(_("%s: user not found\n"), username );
1998-04-20 15:52:20 +00:00
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 */
1998-06-29 12:30:57 +00:00
node = find_kbnode( keyblock, PKT_PUBLIC_KEY );
1998-04-20 15:52:20 +00:00
if( !node ) {
log_error("Oops; public key not found anymore!\n");
rc = G10ERR_GENERAL;
goto leave;
}
1998-06-29 12:30:57 +00:00
pk = node->pkt->pkt.public_key;
keyid_from_pk( pk, pk_keyid );
1998-07-06 10:23:57 +00:00
tty_printf(_("Checking signatures of this public key certificate:\n"));
1998-04-20 15:52:20 +00:00
tty_printf("pub %4u%c/%08lX %s ",
1998-06-29 12:30:57 +00:00
nbits_from_pk( pk ),
pubkey_letter( pk->pubkey_algo ),
pk_keyid[1], datestr_from_pk(pk) );
1998-04-20 15:52:20 +00:00
{
size_t n;
1998-06-29 12:30:57 +00:00
char *p = get_user_id( pk_keyid, &n );
1998-04-20 15:52:20 +00:00
tty_print_string( p, n > 40? 40 : n );
m_free(p);
tty_printf("\n");
}
clear_kbnode_flags( keyblock );
check_all_keysigs( keyblock );
1998-06-29 12:30:57 +00:00
if( remove_keysigs( keyblock, pk_keyid, 1 ) ) {
1998-04-20 15:52:20 +00:00
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;
1998-06-29 12:30:57 +00:00
PKT_public_key *pk = NULL;
PKT_secret_key *sk = NULL;
1998-04-20 15:52:20 +00:00
u32 keyid[2];
int okay=0;
/* search the userid */
rc = secret? find_secret_keyblock_byname( &kbpos, username )
: find_keyblock_byname( &kbpos, username );
if( rc ) {
1998-07-06 10:23:57 +00:00
log_error(_("%s: user not found\n"), username );
1998-04-20 15:52:20 +00:00
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 */
1998-06-29 12:30:57 +00:00
node = find_kbnode( keyblock, secret? PKT_SECRET_KEY:PKT_PUBLIC_KEY );
1998-04-20 15:52:20 +00:00
if( !node ) {
log_error("Oops; key not found anymore!\n");
rc = G10ERR_GENERAL;
goto leave;
}
if( secret ) {
1998-06-29 12:30:57 +00:00
sk = node->pkt->pkt.secret_key;
keyid_from_sk( sk, keyid );
1998-04-20 15:52:20 +00:00
}
else {
1998-06-29 12:30:57 +00:00
pk = node->pkt->pkt.public_key;
keyid_from_pk( pk, keyid );
1998-04-20 15:52:20 +00:00
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 ",
1998-06-29 12:30:57 +00:00
nbits_from_sk( sk ),
pubkey_letter( sk->pubkey_algo ),
keyid[1], datestr_from_sk(sk) );
1998-04-20 15:52:20 +00:00
else
tty_printf("pub %4u%c/%08lX %s ",
1998-06-29 12:30:57 +00:00
nbits_from_pk( pk ),
pubkey_letter( pk->pubkey_algo ),
keyid[1], datestr_from_pk(pk) );
1998-04-20 15:52:20 +00:00
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;
1998-06-29 12:30:57 +00:00
PKT_secret_key *sk;
1998-05-26 13:38:00 +00:00
u32 keyid[2];
1998-04-20 15:52:20 +00:00
char *answer;
int changed=0;
1998-05-26 13:38:00 +00:00
char *passphrase = NULL;
1998-04-20 15:52:20 +00:00
/* 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 */
1998-06-29 12:30:57 +00:00
node = find_kbnode( keyblock, PKT_SECRET_KEY );
1998-04-20 15:52:20 +00:00
if( !node ) {
log_error("Oops; secret key not found anymore!\n");
rc = G10ERR_GENERAL;
goto leave;
}
1998-06-29 12:30:57 +00:00
sk = node->pkt->pkt.secret_key;
keyid_from_sk( sk, keyid );
1998-04-20 15:52:20 +00:00
tty_printf("sec %4u%c/%08lX %s ",
1998-06-29 12:30:57 +00:00
nbits_from_sk( sk ),
pubkey_letter( sk->pubkey_algo ),
keyid[1], datestr_from_sk(sk) );
1998-04-20 15:52:20 +00:00
{
size_t n;
1998-05-26 13:38:00 +00:00
char *p = get_user_id( keyid, &n );
1998-04-20 15:52:20 +00:00
tty_print_string( p, n );
m_free(p);
tty_printf("\n");
}
1998-05-26 13:38:00 +00:00
for(node=keyblock; node; node = node->next ) {
1998-06-29 12:30:57 +00:00
if( node->pkt->pkttype == PKT_SECRET_SUBKEY ) {
PKT_secret_key *subsk = node->pkt->pkt.secret_key;
keyid_from_sk( subsk, keyid );
1998-05-26 13:38:00 +00:00
tty_printf("sub %4u%c/%08lX %s\n",
1998-06-29 12:30:57 +00:00
nbits_from_sk( subsk ),
pubkey_letter( subsk->pubkey_algo ),
keyid[1], datestr_from_sk(subsk) );
1998-05-26 13:38:00 +00:00
}
}
1998-04-20 15:52:20 +00:00
clear_kbnode_flags( keyblock );
1998-06-29 12:30:57 +00:00
switch( is_secret_key_protected( sk ) ) {
1998-04-20 15:52:20 +00:00
case -1:
rc = G10ERR_PUBKEY_ALGO;
break;
case 0:
1998-07-06 10:23:57 +00:00
tty_printf(_("This key is not protected.\n"));
1998-04-20 15:52:20 +00:00
break;
default:
1998-07-06 10:23:57 +00:00
tty_printf(_("Key is protected.\n"));
1998-06-29 12:30:57 +00:00
rc = check_secret_key( sk );
1998-05-26 13:38:00 +00:00
if( !rc )
passphrase = get_last_passphrase();
1998-04-20 15:52:20 +00:00
break;
}
1998-05-26 13:38:00 +00:00
/* unprotect all subkeys (use the supplied passphrase or ask)*/
for(node=keyblock; node; node = node->next ) {
1998-06-29 12:30:57 +00:00
if( node->pkt->pkttype == PKT_SECRET_SUBKEY ) {
PKT_secret_key *subsk = node->pkt->pkt.secret_key;
1998-05-26 13:38:00 +00:00
set_next_passphrase( passphrase );
1998-06-29 12:30:57 +00:00
rc = check_secret_key( subsk );
1998-05-26 13:38:00 +00:00
if( rc )
break;
}
}
1998-05-05 20:34:20 +00:00
1998-04-20 15:52:20 +00:00
if( rc )
1998-07-06 10:23:57 +00:00
tty_printf(_("Can't edit this key: %s\n"), g10_errstr(rc));
1998-04-20 15:52:20 +00:00
else {
1998-05-04 18:49:26 +00:00
DEK *dek = NULL;
1998-05-03 15:42:08 +00:00
STRING2KEY *s2k = m_alloc_secure( sizeof *s2k );
1998-04-20 15:52:20 +00:00
1998-05-04 18:49:26 +00:00
tty_printf(_("Enter the new passphrase for this secret key.\n\n") );
1998-04-20 15:52:20 +00:00
1998-05-26 13:38:00 +00:00
set_next_passphrase( NULL );
1998-04-20 15:52:20 +00:00
for(;;) {
1998-05-03 15:42:08 +00:00
s2k->mode = 1;
s2k->hash_algo = DIGEST_ALGO_RMD160;
1998-05-04 18:49:26 +00:00
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 ) {
1998-04-20 15:52:20 +00:00
rc = 0;
1998-05-04 18:49:26 +00:00
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? "));
1998-04-20 15:52:20 +00:00
tty_kill_prompt();
if( answer_is_yes(answer) )
changed++;
m_free(answer);
break;
}
else { /* okay */
1998-06-29 12:30:57 +00:00
sk->protect.algo = dek->algo;
sk->protect.s2k = *s2k;
rc = protect_secret_key( sk, dek );
1998-05-26 13:38:00 +00:00
for(node=keyblock; !rc && node; node = node->next ) {
1998-06-29 12:30:57 +00:00
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 );
1998-05-26 13:38:00 +00:00
}
}
1998-04-20 15:52:20 +00:00
if( rc )
log_error("protect_secret_key failed: %s\n", g10_errstr(rc) );
else
changed++;
break;
}
}
1998-05-03 15:42:08 +00:00
m_free(s2k);
1998-04-20 15:52:20 +00:00
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:
1998-05-26 13:38:00 +00:00
m_free( passphrase );
1998-04-20 15:52:20 +00:00
release_kbnode( keyblock );
1998-05-26 13:38:00 +00:00
set_next_passphrase( NULL );
1998-04-20 15:52:20 +00:00
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)
1998-05-13 17:53:36 +00:00
* If digest_algo is 0 the function selects an appropriate one.
1998-04-20 15:52:20 +00:00
*/
int
1998-06-29 12:30:57 +00:00
make_keysig_packet( PKT_signature **ret_sig, PKT_public_key *pk,
PKT_user_id *uid, PKT_public_key *subpk,
PKT_secret_key *sk,
1998-06-25 10:19:08 +00:00
int sigclass, int digest_algo,
int (*mksubpkt)(PKT_signature *, void *), void *opaque
)
1998-04-20 15:52:20 +00:00
{
PKT_signature *sig;
int rc=0;
MD_HANDLE md;
1998-05-26 13:38:00 +00:00
assert( (sigclass >= 0x10 && sigclass <= 0x13)
|| sigclass == 0x20 || sigclass == 0x18 );
1998-05-13 17:53:36 +00:00
if( !digest_algo ) {
1998-06-29 12:30:57 +00:00
switch( sk->pubkey_algo ) {
1998-05-13 17:53:36 +00:00
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;
}
}
1998-04-20 15:52:20 +00:00
md = md_open( digest_algo, 0 );
/* hash the public key certificate and the user id */
1998-06-29 12:30:57 +00:00
hash_public_key( md, pk );
1998-05-26 13:38:00 +00:00
if( sigclass == 0x18 ) { /* subkey binding */
1998-06-29 12:30:57 +00:00
hash_public_key( md, subpk );
1998-05-26 13:38:00 +00:00
}
else if( sigclass != 0x20 ) {
1998-06-29 12:30:57 +00:00
if( sk->version >=4 ) {
1998-05-13 17:53:36 +00:00
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 );
}
1998-04-20 15:52:20 +00:00
md_write( md, uid->name, uid->len );
1998-05-13 17:53:36 +00:00
}
1998-04-20 15:52:20 +00:00
/* and make the signature packet */
sig = m_alloc_clear( sizeof *sig );
1998-06-29 12:30:57 +00:00
sig->version = sk->version;
keyid_from_sk( sk, sig->keyid );
sig->pubkey_algo = sk->pubkey_algo;
1998-05-13 17:53:36 +00:00
sig->digest_algo = digest_algo;
1998-04-20 15:52:20 +00:00
sig->timestamp = make_timestamp();
sig->sig_class = sigclass;
1998-06-25 10:19:08 +00:00
if( sig->version >= 4 )
1998-05-13 17:53:36 +00:00
build_sig_subpkt_from_sig( sig );
1998-06-25 10:19:08 +00:00
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 );
1998-05-13 17:53:36 +00:00
}
1998-06-25 10:19:08 +00:00
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 );
1998-05-13 17:53:36 +00:00
1998-06-25 10:19:08 +00:00
}
md_final(md);
1998-04-20 15:52:20 +00:00
1998-06-29 12:30:57 +00:00
rc = complete_sig( sig, sk, md );
1998-06-25 10:19:08 +00:00
}
1998-04-20 15:52:20 +00:00
md_close( md );
if( rc )
free_seckey_enc( sig );
else
*ret_sig = sig;
return rc;
}