ce02c031ed0ea08e4e492d572657a85a7b460956
[gpgme.git] / gpgme / sign.c
1 /* sign.c - Signing function.
2    Copyright (C) 2000 Werner Koch (dd9jn)
3    Copyright (C) 2001, 2002, 2003 g10 Code GmbH
4
5    This file is part of GPGME.
6  
7    GPGME is free software; you can redistribute it and/or modify it
8    under the terms of the GNU General Public License as published by
9    the Free Software Foundation; either version 2 of the License, or
10    (at your option) any later version.
11  
12    GPGME is distributed in the hope that it will be useful, but
13    WITHOUT ANY WARRANTY; without even the implied warranty of
14    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
15    General Public License for more details.
16  
17    You should have received a copy of the GNU General Public License
18    along with GPGME; if not, write to the Free Software Foundation,
19    Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.  */
20
21 #if HAVE_CONFIG_H
22 #include <config.h>
23 #endif
24 #include <stdlib.h>
25 #include <string.h>
26 #include <errno.h>
27
28 #include "gpgme.h"
29 #include "context.h"
30 #include "ops.h"
31
32 \f
33 typedef struct
34 {
35   struct _gpgme_op_sign_result result;
36
37   /* A pointer to the next pointer of the last invalid signer in
38      the list.  This makes appending new invalid signers painless
39      while preserving the order.  */
40   gpgme_invalid_user_id_t *last_signer_p;
41
42   /* Likewise for signature information.  */
43   gpgme_new_signature_t *last_sig_p;
44 } *op_data_t;
45
46
47 static void
48 release_op_data (void *hook)
49 {
50   op_data_t opd = (op_data_t) hook;
51   gpgme_invalid_user_id_t invalid_signer = opd->result.invalid_signers;
52   gpgme_new_signature_t sig = opd->result.signatures;
53
54   while (invalid_signer)
55     {
56       gpgme_invalid_user_id_t next = invalid_signer->next;
57       free (invalid_signer->id);
58       free (invalid_signer);
59       invalid_signer = next;
60     }
61
62   while (sig)
63     {
64       gpgme_new_signature_t next = sig->next;
65       free (sig->fpr);
66       free (sig);
67       sig = next;
68     }
69 }
70
71
72 gpgme_sign_result_t
73 gpgme_op_sign_result (gpgme_ctx_t ctx)
74 {
75   void *hook;
76   op_data_t opd;
77   gpgme_error_t err;
78
79   err = _gpgme_op_data_lookup (ctx, OPDATA_SIGN, &hook, -1, NULL);
80   opd = hook;
81   if (err || !opd)
82     return NULL;
83
84   return &opd->result;
85 }
86
87 \f
88 static gpgme_error_t
89 parse_sig_created (char *args, gpgme_new_signature_t *sigp)
90 {
91   gpgme_new_signature_t sig;
92   char *tail;
93
94   sig = malloc (sizeof (*sig));
95   if (!sig)
96     return GPGME_Out_Of_Core;
97
98   sig->next = NULL;
99   switch (*args)
100     {
101     case 'S':
102       sig->type = GPGME_SIG_MODE_NORMAL;
103       break;
104
105     case 'D':
106       sig->type = GPGME_SIG_MODE_DETACH;
107       break;
108
109     case 'C':
110       sig->type = GPGME_SIG_MODE_CLEAR;
111       break;
112
113     default:
114       /* The backend engine is not behaving.  */
115       free (sig);
116       return GPGME_General_Error;
117     }
118
119   args++;
120   if (*args != ' ')
121     {
122       free (sig);
123       return GPGME_General_Error;
124     }
125
126   errno = 0;
127   sig->pubkey_algo = strtol (args, &tail, 0);
128   if (errno || args == tail || *tail != ' ')
129     {
130       /* The crypto backend does not behave.  */
131       free (sig);
132       return GPGME_General_Error;
133     }
134   args = tail;
135
136   sig->hash_algo = strtol (args, &tail, 0);
137   if (errno || args == tail || *tail != ' ')
138     {
139       /* The crypto backend does not behave.  */
140       free (sig);
141       return GPGME_General_Error;
142     }
143   args = tail;
144
145   sig->class = strtol (args, &tail, 0);
146   if (errno || args == tail || *tail != ' ')
147     {
148       /* The crypto backend does not behave.  */
149       free (sig);
150       return GPGME_General_Error;
151     }
152   args = tail;
153
154   sig->timestamp = strtol (args, &tail, 0);
155   if (errno || args == tail || *tail != ' ')
156     {
157       /* The crypto backend does not behave.  */
158       free (sig);
159       return GPGME_General_Error;
160     }
161   args = tail;
162   while (*args == ' ')
163     args++;
164
165   if (!*args)
166     {
167       /* The crypto backend does not behave.  */
168       free (sig);
169       return GPGME_General_Error;
170     }
171
172   tail = strchr (args, ' ');
173   if (tail)
174     *tail = '\0';
175
176   sig->fpr = strdup (args);
177   if (!sig->fpr)
178     {
179       free (sig);
180       return GPGME_Out_Of_Core;
181     }
182   *sigp = sig;
183   return 0;
184 }
185
186
187 gpgme_error_t
188 _gpgme_sign_status_handler (void *priv, gpgme_status_code_t code, char *args)
189 {
190   gpgme_ctx_t ctx = (gpgme_ctx_t) priv;
191   gpgme_error_t err;
192   void *hook;
193   op_data_t opd;
194
195   err = _gpgme_passphrase_status_handler (priv, code, args);
196   if (err)
197     return err;
198
199   err = _gpgme_op_data_lookup (ctx, OPDATA_SIGN, &hook, -1, NULL);
200   opd = hook;
201   if (err)
202     return err;
203
204   switch (code)
205     {
206     case GPGME_STATUS_SIG_CREATED:
207       err = parse_sig_created (args, opd->last_sig_p);
208       if (err)
209         return err;
210
211       opd->last_sig_p = &(*opd->last_sig_p)->next;
212       break;
213
214     case GPGME_STATUS_INV_RECP:
215       err = _gpgme_parse_inv_userid (args, opd->last_signer_p);
216       if (err)
217         return err;
218
219       opd->last_signer_p = &(*opd->last_signer_p)->next;
220       break;
221
222     case GPGME_STATUS_EOF:
223       if (opd->result.invalid_signers)
224         return GPGME_Invalid_UserID;
225       break;
226
227     default:
228       break;
229     }
230   return err;
231 }
232
233
234 static gpgme_error_t
235 sign_status_handler (void *priv, gpgme_status_code_t code, char *args)
236 {
237   return _gpgme_progress_status_handler (priv, code, args)
238     || _gpgme_sign_status_handler (priv, code, args);
239 }
240
241
242 gpgme_error_t
243 _gpgme_op_sign_init_result (gpgme_ctx_t ctx)
244 {
245   gpgme_error_t err;
246   void *hook;
247   op_data_t opd;
248
249   err = _gpgme_op_data_lookup (ctx, OPDATA_SIGN, &hook,
250                                sizeof (*opd), release_op_data);
251   opd = hook;
252   if (err)
253     return err;
254   opd->last_signer_p = &opd->result.invalid_signers;
255   opd->last_sig_p = &opd->result.signatures;
256   return 0;
257 }
258
259
260 static gpgme_error_t
261 sign_start (gpgme_ctx_t ctx, int synchronous, gpgme_data_t plain,
262             gpgme_data_t sig, gpgme_sig_mode_t mode)
263 {
264   gpgme_error_t err;
265
266   err = _gpgme_op_reset (ctx, synchronous);
267   if (err)
268     return err;
269
270   err = _gpgme_op_sign_init_result (ctx);
271   if (err)
272     return err;
273
274   if (mode != GPGME_SIG_MODE_NORMAL && mode != GPGME_SIG_MODE_DETACH
275       && mode != GPGME_SIG_MODE_CLEAR)
276     return GPGME_Invalid_Value;
277
278   if (!plain)
279     return GPGME_No_Data;
280   if (!sig)
281     return GPGME_Invalid_Value;
282
283   if (ctx->passphrase_cb)
284     {
285       err = _gpgme_engine_set_command_handler
286         (ctx->engine, _gpgme_passphrase_command_handler, ctx, NULL);
287       if (err)
288         return err;
289     }
290
291   _gpgme_engine_set_status_handler (ctx->engine, sign_status_handler,
292                                     ctx);
293
294   return _gpgme_engine_op_sign (ctx->engine, plain, sig, mode, ctx->use_armor,
295                                 ctx->use_textmode, ctx->include_certs,
296                                 ctx /* FIXME */);
297 }
298
299
300 /* Sign the plaintext PLAIN and store the signature in SIG.  */
301 gpgme_error_t
302 gpgme_op_sign_start (gpgme_ctx_t ctx, gpgme_data_t plain, gpgme_data_t sig,
303                      gpgme_sig_mode_t mode)
304 {
305   return sign_start (ctx, 0, plain, sig, mode);
306 }
307
308
309 /* Sign the plaintext PLAIN and store the signature in SIG.  */
310 gpgme_error_t
311 gpgme_op_sign (gpgme_ctx_t ctx, gpgme_data_t plain, gpgme_data_t sig,
312                gpgme_sig_mode_t mode)
313 {
314   gpgme_error_t err = sign_start (ctx, 1, plain, sig, mode);
315   if (!err)
316     err = _gpgme_wait_one (ctx);
317   return err;
318 }