* decrypt.c (is_token,skip_token): Duplicated from verify.c
[gpgme.git] / gpgme / decrypt.c
1 /* decrypt.c -  decrypt functions
2  *      Copyright (C) 2000 Werner Koch (dd9jn)
3  *      Copyright (C) 2001, 2002 g10 Code GmbH
4  *
5  * This file is part of GPGME.
6  *
7  * GPGME is free software; you can redistribute it and/or modify
8  * it 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,
13  * but WITHOUT ANY WARRANTY; without even the implied warranty of
14  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
15  * GNU General Public License for more details.
16  *
17  * You should have received a copy of the GNU General Public License
18  * along with this program; if not, write to the Free Software
19  * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA
20  */
21
22 #include <config.h>
23 #include <stdio.h>
24 #include <stdlib.h>
25 #include <string.h>
26 #include <assert.h>
27
28 #include "util.h"
29 #include "context.h"
30 #include "ops.h"
31
32
33 struct decrypt_result_s
34 {
35   int okay;
36   int failed;
37 };
38
39
40 void
41 _gpgme_release_decrypt_result (DecryptResult result)
42 {
43   if (!result)
44     return;
45   xfree (result);
46 }
47
48 /* Check whether STRING starts with TOKEN and return true in this
49    case.  This is case insensitive.  If NEXT is not NULL return the
50    number of bytes to be added to STRING to get to the next token; a
51    returned value of 0 indicates end of line. 
52    Fixme: Duplicated from verify.c. */
53 static int 
54 is_token (const char *string, const char *token, size_t *next)
55 {
56   size_t n = 0;
57
58   for (;*string && *token && *string == *token; string++, token++, n++)
59     ;
60   if (*token || (*string != ' ' && !*string))
61     return 0;
62   if (next)
63     {
64       for (; *string == ' '; string++, n++)
65         ;
66       *next = n;
67     }
68   return 1;
69 }
70
71 static int
72 skip_token (const char *string, size_t *next)
73 {
74   size_t n = 0;
75
76   for (;*string && *string != ' '; string++, n++)
77     ;
78   for (;*string == ' '; string++, n++)
79     ;
80   if (!*string)
81     return 0;
82   if (next)
83     *next = n;
84   return 1;
85 }
86
87
88 void
89 _gpgme_decrypt_status_handler (GpgmeCtx ctx, GpgmeStatusCode code, char *args)
90 {
91   size_t n;
92
93   _gpgme_passphrase_status_handler (ctx, code, args);
94
95   if (ctx->error)
96     return;
97   test_and_allocate_result (ctx, decrypt);
98
99   switch (code)
100     {
101     case GPGME_STATUS_EOF:
102       if (ctx->result.decrypt->failed)
103         ctx->error = mk_error (Decryption_Failed);
104       else if (!ctx->result.decrypt->okay)
105         ctx->error = mk_error (No_Data);
106       break;
107
108     case GPGME_STATUS_DECRYPTION_OKAY:
109       ctx->result.decrypt->okay = 1;
110       break;
111
112     case GPGME_STATUS_DECRYPTION_FAILED:
113       ctx->result.decrypt->failed = 1;
114       break;
115
116     case GPGME_STATUS_ERROR:
117       if (is_token (args, "decrypt.algorithm", &n) && n)
118         {
119           args += n;
120           if (is_token (args, "Unsupported_Algorithm", &n))
121             {
122               GpgmeData dh;
123
124               args += n;
125               /* Fixme: This won't work when used with decrypt+verify */
126               if (!gpgme_data_new (&dh))
127                 {
128                   _gpgme_data_append_string (dh,
129                                              "<GnupgOperationInfo>\n"
130                                              " <decryption>\n"
131                                              "  <error>\n"
132                                              "   <unsupportedAlgorithm>");
133                   if (skip_token (args, &n))
134                     {
135                       int c = args[n];
136                       args[n] = 0;
137                       _gpgme_data_append_percentstring_for_xml (dh, args);
138                       args[n] = c;
139                     }
140                   else
141                     _gpgme_data_append_percentstring_for_xml (dh, args);
142                   
143                   _gpgme_data_append_string (dh,
144                                              "</unsupportedAlgorithm>\n"
145                                              "  </error>\n"
146                                              " </decryption>\n"
147                                              "</GnupgOperationInfo>\n");
148                   _gpgme_set_op_info (ctx, dh);
149                 }
150             }
151         }
152       break;
153
154         
155     default:
156       /* Ignore all other codes.  */
157       break;
158     }
159 }
160
161
162 GpgmeError
163 _gpgme_decrypt_start (GpgmeCtx ctx, int synchronous,
164                       GpgmeData ciph, GpgmeData plain, void *status_handler)
165 {
166   GpgmeError err = 0;
167
168   err = _gpgme_op_reset (ctx, synchronous);
169   if (err)
170     goto leave;
171
172   /* Check the supplied data.  */
173   if (!ciph || gpgme_data_get_type (ciph) == GPGME_DATA_TYPE_NONE)
174     {
175       err = mk_error (No_Data);
176       goto leave;
177     }
178   _gpgme_data_set_mode (ciph, GPGME_DATA_MODE_OUT);
179
180   if (gpgme_data_get_type (plain) != GPGME_DATA_TYPE_NONE)
181     {
182       err = mk_error (Invalid_Value);
183       goto leave;
184     }
185   _gpgme_data_set_mode (plain, GPGME_DATA_MODE_IN);
186
187   err = _gpgme_passphrase_start (ctx);
188   if (err)
189     goto leave;
190
191   _gpgme_engine_set_status_handler (ctx->engine, status_handler, ctx);
192   _gpgme_engine_set_verbosity (ctx->engine, ctx->verbosity);
193
194   err = _gpgme_engine_op_decrypt (ctx->engine, ciph, plain);
195
196   if (!err)     /* And kick off the process.  */
197     err = _gpgme_engine_start (ctx->engine, ctx);
198
199  leave:
200   if (err)
201     {
202       ctx->pending = 0; 
203       _gpgme_engine_release (ctx->engine);
204       ctx->engine = NULL;
205     }
206   return err;
207 }
208
209
210 GpgmeError
211 gpgme_op_decrypt_start (GpgmeCtx ctx, GpgmeData ciph, GpgmeData plain)
212 {
213   return _gpgme_decrypt_start (ctx, 0, ciph, plain,
214                                _gpgme_decrypt_status_handler);
215 }
216
217
218 /**
219  * gpgme_op_decrypt:
220  * @ctx: The context
221  * @in: ciphertext input
222  * @out: plaintext output
223  * 
224  * This function decrypts @in to @out.
225  * Other parameters are take from the context @ctx.
226  * The function does wait for the result.
227  * 
228  * Return value:  0 on success or an errorcode. 
229  **/
230 GpgmeError
231 gpgme_op_decrypt (GpgmeCtx ctx, GpgmeData in, GpgmeData out)
232 {
233   GpgmeError err = _gpgme_decrypt_start (ctx, 1, in, out,
234                                          _gpgme_decrypt_status_handler);
235   if (!err)
236       err = _gpgme_wait_one (ctx);
237
238   /* Work around the kludge in engine-gpgsm.c */
239   if (err == GPGME_Invalid_Engine && ctx->error)
240     {
241       if (ctx->result.decrypt->failed)
242         err = mk_error (Decryption_Failed);
243       else if (!ctx->result.decrypt->okay)
244         err = mk_error (No_Data);
245     }
246
247   return err;
248 }