|
Packit |
d7e8d0 |
/* genkey.c - Key generation.
|
|
Packit |
d7e8d0 |
* Copyright (C) 2000 Werner Koch (dd9jn)
|
|
Packit |
d7e8d0 |
* Copyright (C) 2001, 2002, 2003, 2004, 2016 g10 Code GmbH
|
|
Packit |
d7e8d0 |
*
|
|
Packit |
d7e8d0 |
* This file is part of GPGME.
|
|
Packit |
d7e8d0 |
*
|
|
Packit |
d7e8d0 |
* GPGME is free software; you can redistribute it and/or modify it
|
|
Packit |
d7e8d0 |
* under the terms of the GNU Lesser General Public License as
|
|
Packit |
d7e8d0 |
* published by the Free Software Foundation; either version 2.1 of
|
|
Packit |
d7e8d0 |
* the License, or (at your option) any later version.
|
|
Packit |
d7e8d0 |
*
|
|
Packit |
d7e8d0 |
* GPGME is distributed in the hope that it will be useful, but
|
|
Packit |
d7e8d0 |
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
Packit |
d7e8d0 |
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
Packit |
d7e8d0 |
* Lesser General Public License for more details.
|
|
Packit |
d7e8d0 |
*
|
|
Packit |
d7e8d0 |
* You should have received a copy of the GNU Lesser General Public
|
|
Packit Service |
30b792 |
* License along with this program; if not, see <https://gnu.org/licenses/>.
|
|
Packit Service |
30b792 |
* SPDX-License-Identifier: LGPL-2.1-or-later
|
|
Packit |
d7e8d0 |
*/
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
#if HAVE_CONFIG_H
|
|
Packit |
d7e8d0 |
#include <config.h>
|
|
Packit |
d7e8d0 |
#endif
|
|
Packit |
d7e8d0 |
#include <stdlib.h>
|
|
Packit |
d7e8d0 |
#include <string.h>
|
|
Packit |
d7e8d0 |
#include <errno.h>
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
#include "gpgme.h"
|
|
Packit |
d7e8d0 |
#include "debug.h"
|
|
Packit |
d7e8d0 |
#include "context.h"
|
|
Packit |
d7e8d0 |
#include "ops.h"
|
|
Packit |
d7e8d0 |
#include "util.h"
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
typedef struct
|
|
Packit |
d7e8d0 |
{
|
|
Packit |
d7e8d0 |
struct _gpgme_op_genkey_result result;
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
/* The error code from a FAILURE status line or 0. */
|
|
Packit |
d7e8d0 |
gpg_error_t failure_code;
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
/* The error code from certain ERROR status lines or 0. */
|
|
Packit |
d7e8d0 |
gpg_error_t error_code;
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
/* Flag to indicate that a UID is to be added. */
|
|
Packit |
d7e8d0 |
gpg_error_t uidmode;
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
/* The key parameters passed to the crypto engine. */
|
|
Packit |
d7e8d0 |
gpgme_data_t key_parameter;
|
|
Packit |
d7e8d0 |
} *op_data_t;
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
static void
|
|
Packit |
d7e8d0 |
release_op_data (void *hook)
|
|
Packit |
d7e8d0 |
{
|
|
Packit |
d7e8d0 |
op_data_t opd = (op_data_t) hook;
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
if (opd->result.fpr)
|
|
Packit |
d7e8d0 |
free (opd->result.fpr);
|
|
Packit |
d7e8d0 |
if (opd->key_parameter)
|
|
Packit |
d7e8d0 |
gpgme_data_release (opd->key_parameter);
|
|
Packit |
d7e8d0 |
}
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
gpgme_genkey_result_t
|
|
Packit |
d7e8d0 |
gpgme_op_genkey_result (gpgme_ctx_t ctx)
|
|
Packit |
d7e8d0 |
{
|
|
Packit |
d7e8d0 |
void *hook;
|
|
Packit |
d7e8d0 |
op_data_t opd;
|
|
Packit |
d7e8d0 |
gpgme_error_t err;
|
|
Packit |
d7e8d0 |
|
|
Packit Service |
30b792 |
TRACE_BEG (DEBUG_CTX, "gpgme_op_genkey_result", ctx, "");
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
err = _gpgme_op_data_lookup (ctx, OPDATA_GENKEY, &hook, -1, NULL);
|
|
Packit |
d7e8d0 |
opd = hook;
|
|
Packit |
d7e8d0 |
if (err || !opd)
|
|
Packit |
d7e8d0 |
{
|
|
Packit Service |
30b792 |
TRACE_SUC ("result=(null)");
|
|
Packit |
d7e8d0 |
return NULL;
|
|
Packit |
d7e8d0 |
}
|
|
Packit |
d7e8d0 |
|
|
Packit Service |
30b792 |
TRACE_LOG ("fpr = %s, %s, %s", opd->result.fpr,
|
|
Packit |
d7e8d0 |
opd->result.primary ? "primary" : "no primary",
|
|
Packit |
d7e8d0 |
opd->result.sub ? "sub" : "no sub");
|
|
Packit |
d7e8d0 |
|
|
Packit Service |
30b792 |
TRACE_SUC ("result=%p", &opd->result);
|
|
Packit |
d7e8d0 |
return &opd->result;
|
|
Packit |
d7e8d0 |
}
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
/* Parse an error status line. Return the error location and the
|
|
Packit |
d7e8d0 |
error code. The function may modify ARGS. */
|
|
Packit |
d7e8d0 |
static char *
|
|
Packit |
d7e8d0 |
parse_error (char *args, gpg_error_t *r_err)
|
|
Packit |
d7e8d0 |
{
|
|
Packit |
d7e8d0 |
char *where = strchr (args, ' ');
|
|
Packit |
d7e8d0 |
char *which;
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
if (where)
|
|
Packit |
d7e8d0 |
{
|
|
Packit |
d7e8d0 |
*where = '\0';
|
|
Packit |
d7e8d0 |
which = where + 1;
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
where = strchr (which, ' ');
|
|
Packit |
d7e8d0 |
if (where)
|
|
Packit |
d7e8d0 |
*where = '\0';
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
where = args;
|
|
Packit |
d7e8d0 |
}
|
|
Packit |
d7e8d0 |
else
|
|
Packit |
d7e8d0 |
{
|
|
Packit |
d7e8d0 |
*r_err = trace_gpg_error (GPG_ERR_INV_ENGINE);
|
|
Packit |
d7e8d0 |
return NULL;
|
|
Packit |
d7e8d0 |
}
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
*r_err = atoi (which);
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
return where;
|
|
Packit |
d7e8d0 |
}
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
static gpgme_error_t
|
|
Packit |
d7e8d0 |
genkey_status_handler (void *priv, gpgme_status_code_t code, char *args)
|
|
Packit |
d7e8d0 |
{
|
|
Packit |
d7e8d0 |
gpgme_ctx_t ctx = (gpgme_ctx_t) priv;
|
|
Packit |
d7e8d0 |
gpgme_error_t err;
|
|
Packit |
d7e8d0 |
void *hook;
|
|
Packit |
d7e8d0 |
op_data_t opd;
|
|
Packit |
d7e8d0 |
char *loc;
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
/* Pipe the status code through the progress status handler. */
|
|
Packit |
d7e8d0 |
err = _gpgme_progress_status_handler (ctx, code, args);
|
|
Packit |
d7e8d0 |
if (err)
|
|
Packit |
d7e8d0 |
return err;
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
err = _gpgme_op_data_lookup (ctx, OPDATA_GENKEY, &hook, -1, NULL);
|
|
Packit |
d7e8d0 |
opd = hook;
|
|
Packit |
d7e8d0 |
if (err)
|
|
Packit |
d7e8d0 |
return err;
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
switch (code)
|
|
Packit |
d7e8d0 |
{
|
|
Packit |
d7e8d0 |
case GPGME_STATUS_KEY_CREATED:
|
|
Packit |
d7e8d0 |
if (args && *args)
|
|
Packit |
d7e8d0 |
{
|
|
Packit |
d7e8d0 |
if (*args == 'B' || *args == 'P')
|
|
Packit |
d7e8d0 |
{
|
|
Packit |
d7e8d0 |
opd->result.primary = 1;
|
|
Packit |
d7e8d0 |
opd->result.uid = 1;
|
|
Packit |
d7e8d0 |
}
|
|
Packit |
d7e8d0 |
if (*args == 'B' || *args == 'S')
|
|
Packit |
d7e8d0 |
opd->result.sub = 1;
|
|
Packit |
d7e8d0 |
if (args[1] == ' ')
|
|
Packit |
d7e8d0 |
{
|
|
Packit |
d7e8d0 |
if (opd->result.fpr)
|
|
Packit |
d7e8d0 |
free (opd->result.fpr);
|
|
Packit |
d7e8d0 |
opd->result.fpr = strdup (&args[2]);
|
|
Packit |
d7e8d0 |
if (!opd->result.fpr)
|
|
Packit |
d7e8d0 |
return gpg_error_from_syserror ();
|
|
Packit |
d7e8d0 |
}
|
|
Packit |
d7e8d0 |
}
|
|
Packit |
d7e8d0 |
break;
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
case GPGME_STATUS_ERROR:
|
|
Packit |
d7e8d0 |
loc = parse_error (args, &err;;
|
|
Packit |
d7e8d0 |
if (!loc)
|
|
Packit |
d7e8d0 |
return err;
|
|
Packit |
d7e8d0 |
if (!opd->error_code)
|
|
Packit |
d7e8d0 |
opd->error_code = err;
|
|
Packit |
d7e8d0 |
break;
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
case GPGME_STATUS_FAILURE:
|
|
Packit |
d7e8d0 |
opd->failure_code = _gpgme_parse_failure (args);
|
|
Packit |
d7e8d0 |
break;
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
case GPGME_STATUS_EOF:
|
|
Packit |
d7e8d0 |
if (opd->error_code)
|
|
Packit |
d7e8d0 |
return opd->error_code;
|
|
Packit |
d7e8d0 |
else if (!opd->uidmode && !opd->result.primary && !opd->result.sub)
|
|
Packit |
d7e8d0 |
return gpg_error (GPG_ERR_GENERAL);
|
|
Packit |
d7e8d0 |
else if (opd->failure_code)
|
|
Packit |
d7e8d0 |
return opd->failure_code;
|
|
Packit |
d7e8d0 |
else if (opd->uidmode == 1)
|
|
Packit |
d7e8d0 |
opd->result.uid = 1; /* We have no status line, thus this hack. */
|
|
Packit |
d7e8d0 |
break;
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
case GPGME_STATUS_INQUIRE_MAXLEN:
|
|
Packit |
d7e8d0 |
if (ctx->status_cb && !ctx->full_status)
|
|
Packit |
d7e8d0 |
{
|
|
Packit |
d7e8d0 |
err = ctx->status_cb (ctx->status_cb_value, "INQUIRE_MAXLEN", args);
|
|
Packit |
d7e8d0 |
if (err)
|
|
Packit |
d7e8d0 |
return err;
|
|
Packit |
d7e8d0 |
}
|
|
Packit |
d7e8d0 |
break;
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
default:
|
|
Packit |
d7e8d0 |
break;
|
|
Packit |
d7e8d0 |
}
|
|
Packit |
d7e8d0 |
return 0;
|
|
Packit |
d7e8d0 |
}
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
static gpgme_error_t
|
|
Packit |
d7e8d0 |
get_key_parameter (const char *parms, gpgme_data_t *key_parameter)
|
|
Packit |
d7e8d0 |
{
|
|
Packit |
d7e8d0 |
const char *content;
|
|
Packit |
d7e8d0 |
const char *attrib;
|
|
Packit |
d7e8d0 |
const char *endtag;
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
/* Extract the key parameter from the XML structure. */
|
|
Packit |
d7e8d0 |
parms = strstr (parms, "
|
|
Packit |
d7e8d0 |
if (!parms)
|
|
Packit |
d7e8d0 |
return gpg_error (GPG_ERR_INV_VALUE);
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
content = strchr (parms, '>');
|
|
Packit |
d7e8d0 |
if (!content)
|
|
Packit |
d7e8d0 |
return gpg_error (GPG_ERR_INV_VALUE);
|
|
Packit |
d7e8d0 |
content++;
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
attrib = strstr (parms, "format=\"internal\"");
|
|
Packit |
d7e8d0 |
if (!attrib || attrib >= content)
|
|
Packit |
d7e8d0 |
return gpg_error (GPG_ERR_INV_VALUE);
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
endtag = strstr (content, "</GnupgKeyParms>");
|
|
Packit Service |
30b792 |
if (!endtag)
|
|
Packit Service |
30b792 |
endtag = content + strlen (content);
|
|
Packit Service |
30b792 |
|
|
Packit |
d7e8d0 |
/* FIXME: Check that there are no control statements inside. */
|
|
Packit Service |
30b792 |
while (content < endtag
|
|
Packit Service |
30b792 |
&& (content[0] == '\n'
|
|
Packit Service |
30b792 |
|| (content[0] == '\r' && content[1] == '\n')))
|
|
Packit |
d7e8d0 |
content++;
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
return gpgme_data_new_from_mem (key_parameter, content,
|
|
Packit |
d7e8d0 |
endtag - content, 1);
|
|
Packit |
d7e8d0 |
}
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
static gpgme_error_t
|
|
Packit |
d7e8d0 |
genkey_start (gpgme_ctx_t ctx, int synchronous, const char *parms,
|
|
Packit |
d7e8d0 |
gpgme_data_t pubkey, gpgme_data_t seckey)
|
|
Packit |
d7e8d0 |
{
|
|
Packit |
d7e8d0 |
gpgme_error_t err;
|
|
Packit |
d7e8d0 |
void *hook;
|
|
Packit |
d7e8d0 |
op_data_t opd;
|
|
Packit |
d7e8d0 |
err = _gpgme_op_reset (ctx, synchronous);
|
|
Packit |
d7e8d0 |
if (err)
|
|
Packit |
d7e8d0 |
return err;
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
err = _gpgme_op_data_lookup (ctx, OPDATA_GENKEY, &hook,
|
|
Packit |
d7e8d0 |
sizeof (*opd), release_op_data);
|
|
Packit |
d7e8d0 |
opd = hook;
|
|
Packit |
d7e8d0 |
if (err)
|
|
Packit |
d7e8d0 |
return err;
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
err = get_key_parameter (parms, &opd->key_parameter);
|
|
Packit |
d7e8d0 |
if (err)
|
|
Packit |
d7e8d0 |
return err;
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
_gpgme_engine_set_status_handler (ctx->engine, genkey_status_handler, ctx);
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
if (ctx->passphrase_cb)
|
|
Packit |
d7e8d0 |
{
|
|
Packit |
d7e8d0 |
err = _gpgme_engine_set_command_handler
|
|
Packit Service |
30b792 |
(ctx->engine, _gpgme_passphrase_command_handler, ctx);
|
|
Packit |
d7e8d0 |
if (err)
|
|
Packit |
d7e8d0 |
return err;
|
|
Packit |
d7e8d0 |
}
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
return _gpgme_engine_op_genkey (ctx->engine,
|
|
Packit |
d7e8d0 |
NULL, NULL, 0, 0, NULL, 0,
|
|
Packit |
d7e8d0 |
opd->key_parameter,
|
|
Packit |
d7e8d0 |
ctx->use_armor? GENKEY_EXTRAFLAG_ARMOR:0,
|
|
Packit |
d7e8d0 |
pubkey, seckey);
|
|
Packit |
d7e8d0 |
}
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
/* Generate a new keypair and add it to the keyring. PUBKEY and
|
|
Packit |
d7e8d0 |
SECKEY should be null for now. PARMS specifies what keys should be
|
|
Packit |
d7e8d0 |
generated. */
|
|
Packit |
d7e8d0 |
gpgme_error_t
|
|
Packit |
d7e8d0 |
gpgme_op_genkey_start (gpgme_ctx_t ctx, const char *parms,
|
|
Packit |
d7e8d0 |
gpgme_data_t pubkey, gpgme_data_t seckey)
|
|
Packit |
d7e8d0 |
{
|
|
Packit |
d7e8d0 |
gpgme_error_t err;
|
|
Packit |
d7e8d0 |
|
|
Packit Service |
30b792 |
TRACE_BEG (DEBUG_CTX, "gpgme_op_genkey_start", ctx,
|
|
Packit |
d7e8d0 |
"pubkey=%p, seckey=%p", pubkey, seckey);
|
|
Packit Service |
30b792 |
TRACE_LOGBUF (parms, parms? strlen (parms):0);
|
|
Packit |
d7e8d0 |
|
|
Packit Service |
30b792 |
if (!ctx || !parms)
|
|
Packit |
d7e8d0 |
return TRACE_ERR (gpg_error (GPG_ERR_INV_ARG));
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
err = genkey_start (ctx, 0, parms, pubkey, seckey);
|
|
Packit |
d7e8d0 |
return TRACE_ERR (err);
|
|
Packit |
d7e8d0 |
}
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
/* Generate a new keypair and add it to the keyring. PUBKEY and
|
|
Packit |
d7e8d0 |
SECKEY should be null for now. PARMS specifies what keys should be
|
|
Packit |
d7e8d0 |
generated. */
|
|
Packit |
d7e8d0 |
gpgme_error_t
|
|
Packit |
d7e8d0 |
gpgme_op_genkey (gpgme_ctx_t ctx, const char *parms, gpgme_data_t pubkey,
|
|
Packit |
d7e8d0 |
gpgme_data_t seckey)
|
|
Packit |
d7e8d0 |
{
|
|
Packit |
d7e8d0 |
gpgme_error_t err;
|
|
Packit |
d7e8d0 |
|
|
Packit Service |
30b792 |
TRACE_BEG (DEBUG_CTX, "gpgme_op_genkey", ctx,
|
|
Packit |
d7e8d0 |
"pubkey=%p, seckey=%p", pubkey, seckey);
|
|
Packit Service |
30b792 |
TRACE_LOGBUF (parms, parms? strlen (parms):0);
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
if (!ctx)
|
|
Packit |
d7e8d0 |
return TRACE_ERR (gpg_error (GPG_ERR_INV_ARG));
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
err = genkey_start (ctx, 1, parms, pubkey, seckey);
|
|
Packit |
d7e8d0 |
if (!err)
|
|
Packit |
d7e8d0 |
err = _gpgme_wait_one (ctx);
|
|
Packit |
d7e8d0 |
return TRACE_ERR (err);
|
|
Packit |
d7e8d0 |
}
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
static gpgme_error_t
|
|
Packit |
d7e8d0 |
createkey_start (gpgme_ctx_t ctx, int synchronous,
|
|
Packit |
d7e8d0 |
const char *userid, const char *algo,
|
|
Packit |
d7e8d0 |
unsigned long reserved, unsigned long expires,
|
|
Packit |
d7e8d0 |
gpgme_key_t anchorkey, unsigned int flags)
|
|
Packit |
d7e8d0 |
{
|
|
Packit |
d7e8d0 |
gpgme_error_t err;
|
|
Packit |
d7e8d0 |
void *hook;
|
|
Packit |
d7e8d0 |
op_data_t opd;
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
err = _gpgme_op_reset (ctx, synchronous);
|
|
Packit |
d7e8d0 |
if (err)
|
|
Packit |
d7e8d0 |
return err;
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
if (reserved || anchorkey || !userid)
|
|
Packit |
d7e8d0 |
return gpg_error (GPG_ERR_INV_ARG);
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
err = _gpgme_op_data_lookup (ctx, OPDATA_GENKEY, &hook,
|
|
Packit |
d7e8d0 |
sizeof (*opd), release_op_data);
|
|
Packit |
d7e8d0 |
opd = hook;
|
|
Packit |
d7e8d0 |
if (err)
|
|
Packit |
d7e8d0 |
return err;
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
_gpgme_engine_set_status_handler (ctx->engine, genkey_status_handler, ctx);
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
if (ctx->passphrase_cb)
|
|
Packit |
d7e8d0 |
{
|
|
Packit |
d7e8d0 |
err = _gpgme_engine_set_command_handler
|
|
Packit Service |
30b792 |
(ctx->engine, _gpgme_passphrase_command_handler, ctx);
|
|
Packit |
d7e8d0 |
if (err)
|
|
Packit |
d7e8d0 |
return err;
|
|
Packit |
d7e8d0 |
}
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
return _gpgme_engine_op_genkey (ctx->engine,
|
|
Packit |
d7e8d0 |
userid, algo, reserved, expires,
|
|
Packit |
d7e8d0 |
anchorkey, flags,
|
|
Packit |
d7e8d0 |
NULL,
|
|
Packit |
d7e8d0 |
ctx->use_armor? GENKEY_EXTRAFLAG_ARMOR:0,
|
|
Packit |
d7e8d0 |
NULL, NULL);
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
}
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
gpgme_error_t
|
|
Packit |
d7e8d0 |
gpgme_op_createkey_start (gpgme_ctx_t ctx, const char *userid, const char *algo,
|
|
Packit |
d7e8d0 |
unsigned long reserved, unsigned long expires,
|
|
Packit |
d7e8d0 |
gpgme_key_t anchorkey, unsigned int flags)
|
|
Packit |
d7e8d0 |
{
|
|
Packit |
d7e8d0 |
gpgme_error_t err;
|
|
Packit |
d7e8d0 |
|
|
Packit Service |
30b792 |
TRACE_BEG (DEBUG_CTX, "gpgme_op_createkey_start", ctx,
|
|
Packit |
d7e8d0 |
"userid='%s', algo='%s' flags=0x%x", userid, algo, flags);
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
if (!ctx)
|
|
Packit |
d7e8d0 |
return TRACE_ERR (gpg_error (GPG_ERR_INV_ARG));
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
err = createkey_start (ctx, 0,
|
|
Packit |
d7e8d0 |
userid, algo, reserved, expires, anchorkey, flags);
|
|
Packit |
d7e8d0 |
return TRACE_ERR (err);
|
|
Packit |
d7e8d0 |
}
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
gpgme_error_t
|
|
Packit |
d7e8d0 |
gpgme_op_createkey (gpgme_ctx_t ctx, const char *userid, const char *algo,
|
|
Packit |
d7e8d0 |
unsigned long reserved, unsigned long expires,
|
|
Packit |
d7e8d0 |
gpgme_key_t anchorkey, unsigned int flags)
|
|
Packit |
d7e8d0 |
{
|
|
Packit |
d7e8d0 |
gpgme_error_t err;
|
|
Packit |
d7e8d0 |
|
|
Packit Service |
30b792 |
TRACE_BEG (DEBUG_CTX, "gpgme_op_createkey", ctx,
|
|
Packit |
d7e8d0 |
"userid='%s', algo='%s' flags=0x%x", userid, algo, flags);
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
if (!ctx)
|
|
Packit |
d7e8d0 |
return TRACE_ERR (gpg_error (GPG_ERR_INV_ARG));
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
err = createkey_start (ctx, 1,
|
|
Packit |
d7e8d0 |
userid, algo, reserved, expires, anchorkey, flags);
|
|
Packit |
d7e8d0 |
if (!err)
|
|
Packit |
d7e8d0 |
err = _gpgme_wait_one (ctx);
|
|
Packit |
d7e8d0 |
return TRACE_ERR (err);
|
|
Packit |
d7e8d0 |
}
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
static gpgme_error_t
|
|
Packit |
d7e8d0 |
createsubkey_start (gpgme_ctx_t ctx, int synchronous,
|
|
Packit |
d7e8d0 |
gpgme_key_t key,
|
|
Packit |
d7e8d0 |
const char *algo,
|
|
Packit |
d7e8d0 |
unsigned long reserved, unsigned long expires,
|
|
Packit |
d7e8d0 |
unsigned int flags)
|
|
Packit |
d7e8d0 |
{
|
|
Packit |
d7e8d0 |
gpgme_error_t err;
|
|
Packit |
d7e8d0 |
void *hook;
|
|
Packit |
d7e8d0 |
op_data_t opd;
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
if (ctx->protocol != GPGME_PROTOCOL_OPENPGP)
|
|
Packit |
d7e8d0 |
return gpgme_error (GPG_ERR_UNSUPPORTED_PROTOCOL);
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
err = _gpgme_op_reset (ctx, synchronous);
|
|
Packit |
d7e8d0 |
if (err)
|
|
Packit |
d7e8d0 |
return err;
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
if (reserved || !key)
|
|
Packit |
d7e8d0 |
return gpg_error (GPG_ERR_INV_ARG);
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
err = _gpgme_op_data_lookup (ctx, OPDATA_GENKEY, &hook,
|
|
Packit |
d7e8d0 |
sizeof (*opd), release_op_data);
|
|
Packit |
d7e8d0 |
opd = hook;
|
|
Packit |
d7e8d0 |
if (err)
|
|
Packit |
d7e8d0 |
return err;
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
_gpgme_engine_set_status_handler (ctx->engine, genkey_status_handler, ctx);
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
if (ctx->passphrase_cb)
|
|
Packit |
d7e8d0 |
{
|
|
Packit |
d7e8d0 |
err = _gpgme_engine_set_command_handler
|
|
Packit Service |
30b792 |
(ctx->engine, _gpgme_passphrase_command_handler, ctx);
|
|
Packit |
d7e8d0 |
if (err)
|
|
Packit |
d7e8d0 |
return err;
|
|
Packit |
d7e8d0 |
}
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
return _gpgme_engine_op_genkey (ctx->engine,
|
|
Packit |
d7e8d0 |
NULL, algo, reserved, expires,
|
|
Packit |
d7e8d0 |
key, flags,
|
|
Packit |
d7e8d0 |
NULL,
|
|
Packit |
d7e8d0 |
ctx->use_armor? GENKEY_EXTRAFLAG_ARMOR:0,
|
|
Packit |
d7e8d0 |
NULL, NULL);
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
}
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
/* Add a subkey to an existing KEY. */
|
|
Packit |
d7e8d0 |
gpgme_error_t
|
|
Packit |
d7e8d0 |
gpgme_op_createsubkey_start (gpgme_ctx_t ctx, gpgme_key_t key, const char *algo,
|
|
Packit |
d7e8d0 |
unsigned long reserved, unsigned long expires,
|
|
Packit |
d7e8d0 |
unsigned int flags)
|
|
Packit |
d7e8d0 |
{
|
|
Packit |
d7e8d0 |
gpgme_error_t err;
|
|
Packit |
d7e8d0 |
|
|
Packit Service |
30b792 |
TRACE_BEG (DEBUG_CTX, "gpgme_op_createsubkey_start", ctx,
|
|
Packit |
d7e8d0 |
"key=%p, algo='%s' flags=0x%x", key, algo, flags);
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
if (!ctx)
|
|
Packit |
d7e8d0 |
return TRACE_ERR (gpg_error (GPG_ERR_INV_ARG));
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
err = createsubkey_start (ctx, 0, key, algo, reserved, expires, flags);
|
|
Packit |
d7e8d0 |
return TRACE_ERR (err);
|
|
Packit |
d7e8d0 |
}
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
gpgme_error_t
|
|
Packit |
d7e8d0 |
gpgme_op_createsubkey (gpgme_ctx_t ctx, gpgme_key_t key, const char *algo,
|
|
Packit |
d7e8d0 |
unsigned long reserved, unsigned long expires,
|
|
Packit |
d7e8d0 |
unsigned int flags)
|
|
Packit |
d7e8d0 |
{
|
|
Packit |
d7e8d0 |
gpgme_error_t err;
|
|
Packit |
d7e8d0 |
|
|
Packit Service |
30b792 |
TRACE_BEG (DEBUG_CTX, "gpgme_op_createsubkey", ctx,
|
|
Packit |
d7e8d0 |
"key=%p, algo='%s' flags=0x%x", key, algo, flags);
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
if (!ctx)
|
|
Packit |
d7e8d0 |
return TRACE_ERR (gpg_error (GPG_ERR_INV_ARG));
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
err = createsubkey_start (ctx, 1, key, algo, reserved, expires, flags);
|
|
Packit |
d7e8d0 |
if (!err)
|
|
Packit |
d7e8d0 |
err = _gpgme_wait_one (ctx);
|
|
Packit |
d7e8d0 |
return TRACE_ERR (err);
|
|
Packit |
d7e8d0 |
}
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
static gpgme_error_t
|
|
Packit |
d7e8d0 |
addrevuid_start (gpgme_ctx_t ctx, int synchronous, int extraflags,
|
|
Packit |
d7e8d0 |
gpgme_key_t key, const char *userid, unsigned int flags)
|
|
Packit |
d7e8d0 |
{
|
|
Packit |
d7e8d0 |
gpgme_error_t err;
|
|
Packit |
d7e8d0 |
void *hook;
|
|
Packit |
d7e8d0 |
op_data_t opd;
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
if (ctx->protocol != GPGME_PROTOCOL_OPENPGP)
|
|
Packit |
d7e8d0 |
return gpgme_error (GPG_ERR_UNSUPPORTED_PROTOCOL);
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
if (!key || !userid)
|
|
Packit |
d7e8d0 |
return gpg_error (GPG_ERR_INV_ARG);
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
err = _gpgme_op_reset (ctx, synchronous);
|
|
Packit |
d7e8d0 |
if (err)
|
|
Packit |
d7e8d0 |
return err;
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
err = _gpgme_op_data_lookup (ctx, OPDATA_GENKEY, &hook,
|
|
Packit |
d7e8d0 |
sizeof (*opd), release_op_data);
|
|
Packit |
d7e8d0 |
opd = hook;
|
|
Packit |
d7e8d0 |
if (err)
|
|
Packit |
d7e8d0 |
return err;
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
opd->uidmode = extraflags? 2 : 1;
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
_gpgme_engine_set_status_handler (ctx->engine, genkey_status_handler, ctx);
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
if (ctx->passphrase_cb)
|
|
Packit |
d7e8d0 |
{
|
|
Packit |
d7e8d0 |
err = _gpgme_engine_set_command_handler
|
|
Packit Service |
30b792 |
(ctx->engine, _gpgme_passphrase_command_handler, ctx);
|
|
Packit |
d7e8d0 |
if (err)
|
|
Packit |
d7e8d0 |
return err;
|
|
Packit |
d7e8d0 |
}
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
return _gpgme_engine_op_genkey (ctx->engine,
|
|
Packit |
d7e8d0 |
userid, NULL, 0, 0,
|
|
Packit |
d7e8d0 |
key, flags,
|
|
Packit |
d7e8d0 |
NULL,
|
|
Packit |
d7e8d0 |
extraflags,
|
|
Packit |
d7e8d0 |
NULL, NULL);
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
}
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
/* Add USERID to an existing KEY. */
|
|
Packit |
d7e8d0 |
gpgme_error_t
|
|
Packit |
d7e8d0 |
gpgme_op_adduid_start (gpgme_ctx_t ctx,
|
|
Packit |
d7e8d0 |
gpgme_key_t key, const char *userid, unsigned int flags)
|
|
Packit |
d7e8d0 |
{
|
|
Packit |
d7e8d0 |
gpgme_error_t err;
|
|
Packit |
d7e8d0 |
|
|
Packit Service |
30b792 |
TRACE_BEG (DEBUG_CTX, "gpgme_op_adduid_start", ctx,
|
|
Packit |
d7e8d0 |
"uid='%s' flags=0x%x", userid, flags);
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
if (!ctx)
|
|
Packit |
d7e8d0 |
return TRACE_ERR (gpg_error (GPG_ERR_INV_ARG));
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
err = addrevuid_start (ctx, 0, 0, key, userid, flags);
|
|
Packit |
d7e8d0 |
return TRACE_ERR (err);
|
|
Packit |
d7e8d0 |
}
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
gpgme_error_t
|
|
Packit |
d7e8d0 |
gpgme_op_adduid (gpgme_ctx_t ctx,
|
|
Packit |
d7e8d0 |
gpgme_key_t key, const char *userid, unsigned int flags)
|
|
Packit |
d7e8d0 |
{
|
|
Packit |
d7e8d0 |
gpgme_error_t err;
|
|
Packit |
d7e8d0 |
|
|
Packit Service |
30b792 |
TRACE_BEG (DEBUG_CTX, "gpgme_op_adduid", ctx,
|
|
Packit |
d7e8d0 |
"uid='%s' flags=0x%x", userid, flags);
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
if (!ctx)
|
|
Packit |
d7e8d0 |
return TRACE_ERR (gpg_error (GPG_ERR_INV_ARG));
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
err = addrevuid_start (ctx, 1, 0, key, userid, flags);
|
|
Packit |
d7e8d0 |
if (!err)
|
|
Packit |
d7e8d0 |
err = _gpgme_wait_one (ctx);
|
|
Packit |
d7e8d0 |
return TRACE_ERR (err);
|
|
Packit |
d7e8d0 |
}
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
/* Revoke USERID from KEY. */
|
|
Packit |
d7e8d0 |
gpgme_error_t
|
|
Packit |
d7e8d0 |
gpgme_op_revuid_start (gpgme_ctx_t ctx,
|
|
Packit |
d7e8d0 |
gpgme_key_t key, const char *userid, unsigned int flags)
|
|
Packit |
d7e8d0 |
{
|
|
Packit |
d7e8d0 |
gpgme_error_t err;
|
|
Packit |
d7e8d0 |
|
|
Packit Service |
30b792 |
TRACE_BEG (DEBUG_CTX, "gpgme_op_revuid_start", ctx,
|
|
Packit |
d7e8d0 |
"uid='%s' flags=0x%x", userid, flags);
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
if (!ctx)
|
|
Packit |
d7e8d0 |
return TRACE_ERR (gpg_error (GPG_ERR_INV_ARG));
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
err = addrevuid_start (ctx, 0, GENKEY_EXTRAFLAG_REVOKE, key, userid, flags);
|
|
Packit |
d7e8d0 |
return TRACE_ERR (err);
|
|
Packit |
d7e8d0 |
}
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
gpgme_error_t
|
|
Packit |
d7e8d0 |
gpgme_op_revuid (gpgme_ctx_t ctx,
|
|
Packit |
d7e8d0 |
gpgme_key_t key, const char *userid, unsigned int flags)
|
|
Packit |
d7e8d0 |
{
|
|
Packit |
d7e8d0 |
gpgme_error_t err;
|
|
Packit |
d7e8d0 |
|
|
Packit Service |
30b792 |
TRACE_BEG (DEBUG_CTX, "gpgme_op_revuid", ctx,
|
|
Packit |
d7e8d0 |
"uid='%s' flags=0x%x", userid, flags);
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
if (!ctx)
|
|
Packit |
d7e8d0 |
return TRACE_ERR (gpg_error (GPG_ERR_INV_ARG));
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
err = addrevuid_start (ctx, 1, GENKEY_EXTRAFLAG_REVOKE, key, userid, flags);
|
|
Packit |
d7e8d0 |
if (!err)
|
|
Packit |
d7e8d0 |
err = _gpgme_wait_one (ctx);
|
|
Packit |
d7e8d0 |
return TRACE_ERR (err);
|
|
Packit |
d7e8d0 |
}
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
/* Set a flag on the USERID of KEY. The only supported flag right now
|
|
Packit |
d7e8d0 |
* is "primary" to mark the primary key. */
|
|
Packit |
d7e8d0 |
static gpg_error_t
|
|
Packit |
d7e8d0 |
set_uid_flag (gpgme_ctx_t ctx, int synchronous,
|
|
Packit |
d7e8d0 |
gpgme_key_t key, const char *userid,
|
|
Packit |
d7e8d0 |
const char *name, const char *value)
|
|
Packit |
d7e8d0 |
{
|
|
Packit |
d7e8d0 |
gpgme_error_t err;
|
|
Packit |
d7e8d0 |
|
|
Packit Service |
30b792 |
TRACE_BEG (DEBUG_CTX, "gpgme_op_set_uid_flag", ctx,
|
|
Packit |
d7e8d0 |
"%d uid='%s' '%s'='%s'", synchronous, userid, name, value);
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
if (!ctx || !name || !key || !userid)
|
|
Packit |
d7e8d0 |
return TRACE_ERR (gpg_error (GPG_ERR_INV_ARG));
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
if (!strcmp (name, "primary"))
|
|
Packit |
d7e8d0 |
{
|
|
Packit |
d7e8d0 |
if (value)
|
|
Packit |
d7e8d0 |
err = gpg_error (GPG_ERR_INV_ARG);
|
|
Packit |
d7e8d0 |
else
|
|
Packit |
d7e8d0 |
err = addrevuid_start (ctx, synchronous,
|
|
Packit |
d7e8d0 |
GENKEY_EXTRAFLAG_SETPRIMARY, key, userid, 0);
|
|
Packit |
d7e8d0 |
}
|
|
Packit |
d7e8d0 |
else
|
|
Packit |
d7e8d0 |
return err = gpg_error (GPG_ERR_UNKNOWN_NAME);
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
if (synchronous && !err)
|
|
Packit |
d7e8d0 |
err = _gpgme_wait_one (ctx);
|
|
Packit |
d7e8d0 |
return TRACE_ERR (err);
|
|
Packit |
d7e8d0 |
}
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
/* See set_uid_flag. */
|
|
Packit |
d7e8d0 |
gpgme_error_t
|
|
Packit |
d7e8d0 |
gpgme_op_set_uid_flag_start (gpgme_ctx_t ctx,
|
|
Packit |
d7e8d0 |
gpgme_key_t key, const char *userid,
|
|
Packit |
d7e8d0 |
const char *name, const char *value)
|
|
Packit |
d7e8d0 |
{
|
|
Packit |
d7e8d0 |
return set_uid_flag (ctx, 0, key, userid, name, value);
|
|
Packit |
d7e8d0 |
}
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
/* See set_uid_flag. This is the synchronous variant. */
|
|
Packit |
d7e8d0 |
gpgme_error_t
|
|
Packit |
d7e8d0 |
gpgme_op_set_uid_flag (gpgme_ctx_t ctx,
|
|
Packit |
d7e8d0 |
gpgme_key_t key, const char *userid,
|
|
Packit |
d7e8d0 |
const char *name, const char *value)
|
|
Packit |
d7e8d0 |
{
|
|
Packit |
d7e8d0 |
return set_uid_flag (ctx, 1, key, userid, name, value);
|
|
Packit |
d7e8d0 |
}
|