2002-08-30 Marcus Brinkmann <marcus@g10code.de>
[gpgme.git] / gpgme / edit.c
1 /* edit.c - key edit functions
2  *      Copyright (C) 2002 g10 Code GmbH
3  *
4  * This file is part of GPGME.
5  *
6  * GPGME is free software; you can redistribute it and/or modify
7  * it under the terms of the GNU General Public License as published by
8  * the Free Software Foundation; either version 2 of the License, or
9  * (at your option) any later version.
10  *
11  * GPGME is distributed in the hope that it will be useful,
12  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14  * GNU General Public License for more details.
15  *
16  * You should have received a copy of the GNU General Public License
17  * along with this program; if not, write to the Free Software
18  * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA
19  */
20
21 #include <config.h>
22 #include <stdio.h>
23 #include <stdlib.h>
24 #include <string.h>
25 #include <assert.h>
26
27 #include "util.h"
28 #include "context.h"
29 #include "ops.h"
30
31
32 struct edit_result_s
33 {
34   GpgmeEditCb fnc;
35   void *fnc_value;
36 };
37
38 void
39 _gpgme_release_edit_result (EditResult result)
40 {
41   if (!result)
42     return;
43   xfree (result);
44 }
45
46 void
47 _gpgme_edit_status_handler (GpgmeCtx ctx, GpgmeStatusCode status, char *args)
48 {
49   _gpgme_passphrase_status_handler (ctx, status, args);
50
51   if (ctx->error)
52     return;
53
54   ctx->error = (*ctx->result.edit->fnc) (ctx->result.edit->fnc_value, status, args, NULL);
55 }
56
57 static const char *
58 command_handler (void *opaque, GpgmeStatusCode status, const char *args)
59 {
60   GpgmeCtx ctx = opaque;
61   const char *result;
62
63   result = _gpgme_passphrase_command_handler (ctx, status, args);
64
65   if (!result)
66     ctx->error = (*ctx->result.edit->fnc) (ctx->result.edit->fnc_value, status, args, &result);
67
68   return result;
69 }
70
71 static GpgmeError
72 _gpgme_op_edit_start (GpgmeCtx ctx, int synchronous,
73                       GpgmeKey key,
74                       GpgmeEditCb fnc, void *fnc_value,
75                       GpgmeData out)
76 {
77   GpgmeError err = 0;
78
79   if (!fnc)
80     return mk_error (Invalid_Value);
81
82   err = _gpgme_op_reset (ctx, synchronous);
83   if (err)
84     goto leave;
85
86   assert (!ctx->result.edit);
87   ctx->result.edit = xtrymalloc (sizeof *ctx->result.edit);
88   if (!ctx->result.edit)
89     {
90       err = mk_error (Out_Of_Core);
91       goto leave;
92     }
93   ctx->result.edit->fnc = fnc;
94   ctx->result.edit->fnc_value = fnc_value;
95
96   /* Check the supplied data.  */
97   if (!out || gpgme_data_get_type (out) != GPGME_DATA_TYPE_NONE)
98     {
99       err = mk_error (Invalid_Value);
100       goto leave;
101     }
102   _gpgme_data_set_mode (out, GPGME_DATA_MODE_IN);
103
104   err = _gpgme_engine_set_command_handler (ctx->engine, command_handler,
105                                            ctx, out);
106   if (err)
107     goto leave;
108
109   _gpgme_engine_set_status_handler (ctx->engine, _gpgme_edit_status_handler,
110                                     ctx);
111
112   _gpgme_engine_set_verbosity (ctx->engine, ctx->verbosity);
113
114   _gpgme_engine_op_edit (ctx->engine, key, out, ctx);
115
116   /* And kick off the process.  */
117   err = _gpgme_engine_start (ctx->engine, ctx);
118   
119  leave:
120   if (err)
121     {
122       ctx->pending = 0; 
123       _gpgme_engine_release (ctx->engine);
124       ctx->engine = NULL;
125     }
126   return err;
127 }
128
129 GpgmeError
130 gpgme_op_edit_start (GpgmeCtx ctx,
131                      GpgmeKey key,
132                      GpgmeEditCb fnc, void *fnc_value,
133                      GpgmeData out)
134 {
135   return _gpgme_op_edit_start (ctx, 0, key, fnc, fnc_value, out);
136 }
137
138 /**
139  * gpgme_op_edit:
140  * @ctx: The context
141  * @key: The key to be edited.
142  * @fnc: An edit callback handler.
143  * @fnc_value: To be passed to @fnc as first arg.
144  * @out: The output.
145  * 
146  * Return value: 0 on success or an error code.
147  **/
148 GpgmeError
149 gpgme_op_edit (GpgmeCtx ctx,
150                GpgmeKey key,
151                GpgmeEditCb fnc, void *fnc_value,
152                GpgmeData out)
153 {
154   GpgmeError err = _gpgme_op_edit_start (ctx, 1, key, fnc, fnc_value, out);
155   if (!err)
156     err = _gpgme_wait_one (ctx);
157   return err;
158 }