2004-06-23 Marcus Brinkmann <marcus@g10code.de>
[gpgme.git] / gpgme / encrypt-sign.c
index 3ea6f28..ae4b594 100644 (file)
 /* encrypt-sign.c -  encrypt and verify functions
- *     Copyright (C) 2000 Werner Koch (dd9jn)
- *      Copyright (C) 2001, 2002 g10 Code GmbH
- *
- * This file is part of GPGME.
- *
- * GPGME 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.
- *
- * GPGME 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
- */
-
-#ifdef HAVE_CONFIG_H
+   Copyright (C) 2000 Werner Koch (dd9jn)
+   Copyright (C) 2001, 2002, 2003 g10 Code GmbH
+   This file is part of GPGME.
+   GPGME 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.
+   GPGME 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 GPGME; if not, write to the Free Software Foundation,
+   Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.  */
+
+#if HAVE_CONFIG_H
 #include <config.h>
 #endif
-#include <stddef.h>
-#include <stdio.h>
-#include <stdlib.h>
-#include <string.h>
-#include <assert.h>
 
-#include "util.h"
+#include "gpgme.h"
 #include "context.h"
 #include "ops.h"
 
-
-static void
-encrypt_sign_status_handler (GpgmeCtx ctx, GpgmeStatusCode code, char *args)
+\f
+static gpgme_error_t
+encrypt_sign_status_handler (void *priv, gpgme_status_code_t code, char *args)
 {
-  char *encrypt_info = 0;
-  size_t encrypt_info_len;
-
-  _gpgme_encrypt_status_handler (ctx, code, args);
+  gpgme_error_t err;
 
-  if (code == GPGME_STATUS_EOF)
-    {
-      encrypt_info = gpgme_data_release_and_get_mem (ctx->op_info,
-                                                    &encrypt_info_len);
-      ctx->op_info = NULL;
-    }
-  _gpgme_sign_status_handler (ctx, code, args);
-  if (code == GPGME_STATUS_EOF && encrypt_info)
-    _gpgme_data_append (ctx->op_info, encrypt_info, encrypt_info_len);
+  err = _gpgme_progress_status_handler (priv, code, args);
+  if (!err)
+    err = _gpgme_encrypt_status_handler (priv, code, args);
+  if (!err)
+    err = _gpgme_sign_status_handler (priv, code, args);
+  return err;
 }
 
 
-static GpgmeError
-_gpgme_op_encrypt_sign_start (GpgmeCtx ctx, int synchronous,
-                             GpgmeRecipients recp,
-                             GpgmeData plain, GpgmeData cipher)
+static gpgme_error_t
+encrypt_sign_start (gpgme_ctx_t ctx, int synchronous, gpgme_key_t recp[],
+                   gpgme_encrypt_flags_t flags,
+                   gpgme_data_t plain, gpgme_data_t cipher)
 {
-  GpgmeError err = 0;
+  gpgme_error_t err;
 
   err = _gpgme_op_reset (ctx, synchronous);
   if (err)
-    goto leave;
+    return err;
 
-  err = _gpgme_passphrase_start (ctx);
+  if (!plain)
+    return gpg_error (GPG_ERR_NO_DATA);
+  if (!cipher || !recp)
+    return gpg_error (GPG_ERR_INV_VALUE);
+
+  err = _gpgme_op_encrypt_init_result (ctx);
   if (err)
-    goto leave;
+    return err;
 
-  _gpgme_engine_set_status_handler (ctx->engine,
-                                   encrypt_sign_status_handler, ctx);
-  _gpgme_engine_set_verbosity (ctx->engine, ctx->verbosity);
+  err = _gpgme_op_sign_init_result (ctx);
+  if (err)
+    return err;
 
-  /* Check the supplied data */
-  if (gpgme_data_get_type (plain) == GPGME_DATA_TYPE_NONE)
-    {
-      err = mk_error (No_Data);
-      goto leave;
-    }
-  _gpgme_data_set_mode (plain, GPGME_DATA_MODE_OUT);
-  if (!cipher || gpgme_data_get_type (cipher) != GPGME_DATA_TYPE_NONE)
+  if (ctx->passphrase_cb)
     {
-      err = mk_error (Invalid_Value);
-      goto leave;
+      err = _gpgme_engine_set_command_handler
+       (ctx->engine, _gpgme_passphrase_command_handler, ctx, NULL);
+      if (err)
+       return err;
     }
-  _gpgme_data_set_mode (cipher, GPGME_DATA_MODE_IN);
 
-  err = _gpgme_engine_op_encrypt_sign (ctx->engine, recp, plain, cipher,
-                                      ctx->use_armor, ctx /* FIXME */);
-
-  if (!err)    /* And kick off the process.  */
-    err = _gpgme_engine_start (ctx->engine, ctx);
-
- leave:
-  if (err)
-    {
-      ctx->pending = 0; 
-      _gpgme_engine_release (ctx->engine);
-      ctx->engine = NULL;
-    }
-  return err;
+  _gpgme_engine_set_status_handler (ctx->engine,
+                                   encrypt_sign_status_handler, ctx);
+  
+  return _gpgme_engine_op_encrypt_sign (ctx->engine, recp, flags, plain,
+                                       cipher, ctx->use_armor,
+                                       ctx /* FIXME */);
 }
 
-GpgmeError
-gpgme_op_encrypt_sign_start (GpgmeCtx ctx, GpgmeRecipients recp,
-                             GpgmeData plain, GpgmeData cipher)
+
+/* Encrypt plaintext PLAIN within CTX for the recipients RECP and
+   store the resulting ciphertext in CIPHER.  Also sign the ciphertext
+   with the signers in CTX.  */
+gpgme_error_t
+gpgme_op_encrypt_sign_start (gpgme_ctx_t ctx, gpgme_key_t recp[],
+                            gpgme_encrypt_flags_t flags,
+                            gpgme_data_t plain, gpgme_data_t cipher)
 {
-  return _gpgme_op_encrypt_sign_start (ctx, 0, recp, plain, cipher);
+  return encrypt_sign_start (ctx, 0, recp, flags, plain, cipher);
 }
 
 
-/**
- * gpgme_op_encrypt_sign:
- * @ctx: The context
- * @recp: The set of recipients
- * @plain: plaintext input
- * @cipher: signed ciphertext
- * 
- * This function encrypts @plain for all recipients in recp, signs it,
- * and returns the ciphertext in @out.  The function does wait for the
- * result.
- * 
- * Return value:  0 on success or an errorcode. 
- **/
-GpgmeError
-gpgme_op_encrypt_sign (GpgmeCtx ctx, GpgmeRecipients recp,
-                      GpgmeData plain, GpgmeData cipher)
+/* Encrypt plaintext PLAIN within CTX for the recipients RECP and
+   store the resulting ciphertext in CIPHER.  Also sign the ciphertext
+   with the signers in CTX.  */
+gpgme_error_t
+gpgme_op_encrypt_sign (gpgme_ctx_t ctx, gpgme_key_t recp[],
+                      gpgme_encrypt_flags_t flags,
+                      gpgme_data_t plain, gpgme_data_t cipher)
 {
-  GpgmeError err = _gpgme_op_encrypt_sign_start (ctx, 1, recp, plain, cipher);
-
+  gpgme_error_t err = encrypt_sign_start (ctx, 1, recp, flags, plain, cipher);
   if (!err)
-    {
-      err = _gpgme_wait_one (ctx);
-      /* Old gpg versions don't return status info for invalid
-         recipients, so we simply check whether we got any output at
-         all, and if not we assume that we don't have valid
-         recipients.  */
-      if (!ctx->error && gpgme_data_get_type (cipher) == GPGME_DATA_TYPE_NONE)
-        ctx->error = mk_error (No_Recipients);
-      err = ctx->error;
-    }
+    err = _gpgme_wait_one (ctx);
   return err;
 }