Blame src/passwd.c

Packit d7e8d0
/* passwd.c - Passphrase changing function
Packit d7e8d0
   Copyright (C) 2010 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 d7e8d0
   License along with this program; if not, see <https://www.gnu.org/licenses/>.
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
Packit d7e8d0
#include "gpgme.h"
Packit d7e8d0
#include "debug.h"
Packit d7e8d0
#include "context.h"
Packit d7e8d0
#include "ops.h"
Packit d7e8d0
Packit d7e8d0
Packit d7e8d0
typedef struct
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
  int success_seen;
Packit d7e8d0
  int error_seen;
Packit d7e8d0
} *op_data_t;
Packit d7e8d0
Packit d7e8d0
Packit d7e8d0

Packit d7e8d0
/* Parse an error status line and return the error code.  */
Packit d7e8d0
static gpgme_error_t
Packit d7e8d0
parse_error (char *args)
Packit d7e8d0
{
Packit d7e8d0
  gpgme_error_t err;
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
    return trace_gpg_error (GPG_ERR_INV_ENGINE);
Packit d7e8d0
Packit d7e8d0
  err = atoi (which);
Packit d7e8d0
Packit d7e8d0
  if (!strcmp (where, "keyedit.passwd"))
Packit d7e8d0
    return err;
Packit d7e8d0
Packit d7e8d0
  return 0;
Packit d7e8d0
}
Packit d7e8d0
Packit d7e8d0
Packit d7e8d0
static gpgme_error_t
Packit d7e8d0
passwd_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
Packit d7e8d0
  err = _gpgme_op_data_lookup (ctx, OPDATA_PASSWD, &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_ERROR:
Packit d7e8d0
      err = parse_error (args);
Packit d7e8d0
      if (err)
Packit d7e8d0
        opd->error_seen = 1;
Packit d7e8d0
      break;
Packit d7e8d0
Packit d7e8d0
    case GPGME_STATUS_SUCCESS:
Packit d7e8d0
      opd->success_seen = 1;
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
      /* In case the OpenPGP engine does not properly implement the
Packit d7e8d0
         passwd command we won't get a success status back and thus we
Packit d7e8d0
         conclude that this operation is not supported.  This is for
Packit d7e8d0
         example the case for GnuPG < 2.0.16.  Note that this test is
Packit d7e8d0
         obsolete for assuan based engines because they will properly
Packit d7e8d0
         return an error for an unknown command.  */
Packit d7e8d0
      if (ctx->protocol == GPGME_PROTOCOL_OpenPGP
Packit d7e8d0
          && !opd->error_seen && !opd->success_seen)
Packit d7e8d0
        err = gpg_error (GPG_ERR_NOT_SUPPORTED);
Packit d7e8d0
      else if (opd->failure_code)
Packit d7e8d0
        err = opd->failure_code;
Packit d7e8d0
      break;
Packit d7e8d0
Packit d7e8d0
    default:
Packit d7e8d0
      break;
Packit d7e8d0
    }
Packit d7e8d0
Packit d7e8d0
  return err;
Packit d7e8d0
}
Packit d7e8d0
Packit d7e8d0
Packit d7e8d0
static gpgme_error_t
Packit d7e8d0
passwd_start (gpgme_ctx_t ctx, int synchronous, gpgme_key_t key,
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 (!key)
Packit d7e8d0
    return gpg_error (GPG_ERR_INV_VALUE);
Packit d7e8d0
  if (flags)
Packit d7e8d0
    return gpg_error (GPG_ERR_INV_FLAG);
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_PASSWD, &hook, sizeof (*opd), NULL);
Packit d7e8d0
  opd = hook;
Packit d7e8d0
  if (err)
Packit d7e8d0
    return err;
Packit d7e8d0
Packit d7e8d0
  opd->success_seen = 0;
Packit d7e8d0
  opd->error_seen = 0;
Packit d7e8d0
Packit d7e8d0
  _gpgme_engine_set_status_handler (ctx->engine, passwd_status_handler, ctx);
Packit d7e8d0
Packit d7e8d0
  if (ctx->passphrase_cb)
Packit d7e8d0
    {
Packit d7e8d0
      err = _gpgme_engine_set_command_handler
Packit d7e8d0
        (ctx->engine, _gpgme_passphrase_command_handler, ctx, NULL);
Packit d7e8d0
      if (err)
Packit d7e8d0
        return err;
Packit d7e8d0
    }
Packit d7e8d0
Packit d7e8d0
  return _gpgme_engine_op_passwd (ctx->engine, key, flags);
Packit d7e8d0
}
Packit d7e8d0
Packit d7e8d0
Packit d7e8d0
Packit d7e8d0
/* Change the passphrase for KEY.  FLAGS is reserved for future use
Packit d7e8d0
   and must be passed as 0.  The engine is expected to present a user
Packit d7e8d0
   interface to enter the old and the new passphrase.  This is the
Packit d7e8d0
   asynchronous variant.
Packit d7e8d0
Packit d7e8d0
   Note that if ever the need arises to supply a passphrase we can do
Packit d7e8d0
   this with a flag value and the passphrase callback feature.  */
Packit d7e8d0
gpgme_error_t
Packit d7e8d0
gpgme_op_passwd_start (gpgme_ctx_t ctx, gpgme_key_t key, unsigned int flags)
Packit d7e8d0
{
Packit d7e8d0
  gpg_error_t err;
Packit d7e8d0
  TRACE_BEG2 (DEBUG_CTX, "gpgme_op_passwd_start", ctx,
Packit d7e8d0
	      "key=%p, flags=0x%x", key, flags);
Packit d7e8d0
Packit d7e8d0
  if (!ctx)
Packit d7e8d0
    return TRACE_ERR (gpg_error (GPG_ERR_INV_VALUE));
Packit d7e8d0
Packit d7e8d0
  err = passwd_start (ctx, 0, key, flags);
Packit d7e8d0
  return TRACE_ERR (err);
Packit d7e8d0
}
Packit d7e8d0
Packit d7e8d0
Packit d7e8d0
/* Change the passphrase for KEY.  FLAGS is reserved for future use
Packit d7e8d0
   and must be passed as 0.  This is the synchronous variant.  */
Packit d7e8d0
gpgme_error_t
Packit d7e8d0
gpgme_op_passwd (gpgme_ctx_t ctx, gpgme_key_t key, unsigned int flags)
Packit d7e8d0
{
Packit d7e8d0
  gpgme_error_t err;
Packit d7e8d0
Packit d7e8d0
  TRACE_BEG2 (DEBUG_CTX, "gpgme_op_passwd", ctx,
Packit d7e8d0
	      "key=%p, flags=0x%x", key, flags);
Packit d7e8d0
Packit d7e8d0
  if (!ctx)
Packit d7e8d0
    return TRACE_ERR (gpg_error (GPG_ERR_INV_VALUE));
Packit d7e8d0
Packit d7e8d0
  err = passwd_start (ctx, 1, key, flags);
Packit d7e8d0
  if (!err)
Packit d7e8d0
    err = _gpgme_wait_one (ctx);
Packit d7e8d0
  return TRACE_ERR (err);
Packit d7e8d0
}
Packit d7e8d0