2003-01-30 Marcus Brinkmann <marcus@g10code.de>
[gpgme.git] / gpgme / decrypt.c
1 /* decrypt.c - Decrypt 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
26 #include "util.h"
27 #include "context.h"
28 #include "ops.h"
29
30
31 struct decrypt_result_s
32 {
33   int okay;
34   int failed;
35 };
36
37
38 void
39 _gpgme_release_decrypt_result (DecryptResult result)
40 {
41   if (!result)
42     return;
43   free (result);
44 }
45
46 /* Check whether STRING starts with TOKEN and return true in this
47    case.  This is case insensitive.  If NEXT is not NULL return the
48    number of bytes to be added to STRING to get to the next token; a
49    returned value of 0 indicates end of line. 
50    Fixme: Duplicated from verify.c.  */
51 static int 
52 is_token (const char *string, const char *token, size_t *next)
53 {
54   size_t n = 0;
55
56   for (;*string && *token && *string == *token; string++, token++, n++)
57     ;
58   if (*token || (*string != ' ' && !*string))
59     return 0;
60   if (next)
61     {
62       for (; *string == ' '; string++, n++)
63         ;
64       *next = n;
65     }
66   return 1;
67 }
68
69
70 static int
71 skip_token (const char *string, size_t *next)
72 {
73   size_t n = 0;
74
75   for (;*string && *string != ' '; string++, n++)
76     ;
77   for (;*string == ' '; string++, n++)
78     ;
79   if (!*string)
80     return 0;
81   if (next)
82     *next = n;
83   return 1;
84 }
85
86
87 GpgmeError
88 _gpgme_decrypt_status_handler (GpgmeCtx ctx, GpgmeStatusCode code, char *args)
89 {
90   GpgmeError err;
91   size_t n;
92
93   err = _gpgme_passphrase_status_handler (ctx, code, args);
94   if (err)
95     return err;
96
97   test_and_allocate_result (ctx, decrypt);
98
99   switch (code)
100     {
101     case GPGME_STATUS_EOF:
102       if (ctx->result.decrypt->failed)
103         return GPGME_Decryption_Failed;
104       else if (!ctx->result.decrypt->okay)
105         return GPGME_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     default:
155       break;
156     }
157
158   return 0;
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)
174     {
175       err = GPGME_No_Data;
176       goto leave;
177     }
178   if (!plain)
179     {
180       err = GPGME_Invalid_Value;
181       goto leave;
182     }
183
184   err = _gpgme_passphrase_start (ctx);
185   if (err)
186     goto leave;
187
188   _gpgme_engine_set_status_handler (ctx->engine, status_handler, ctx);
189   _gpgme_engine_set_verbosity (ctx->engine, ctx->verbosity);
190
191   err = _gpgme_engine_op_decrypt (ctx->engine, ciph, plain);
192
193  leave:
194   if (err)
195     {
196       ctx->pending = 0; 
197       _gpgme_engine_release (ctx->engine);
198       ctx->engine = NULL;
199     }
200   return err;
201 }
202
203
204 GpgmeError
205 gpgme_op_decrypt_start (GpgmeCtx ctx, GpgmeData ciph, GpgmeData plain)
206 {
207   return _gpgme_decrypt_start (ctx, 0, ciph, plain,
208                                _gpgme_decrypt_status_handler);
209 }
210
211
212 /**
213  * gpgme_op_decrypt:
214  * @ctx: The context
215  * @in: ciphertext input
216  * @out: plaintext output
217  * 
218  * This function decrypts @in to @out.
219  * Other parameters are take from the context @ctx.
220  * The function does wait for the result.
221  * 
222  * Return value:  0 on success or an errorcode. 
223  **/
224 GpgmeError
225 gpgme_op_decrypt (GpgmeCtx ctx, GpgmeData in, GpgmeData out)
226 {
227   GpgmeError err = _gpgme_decrypt_start (ctx, 1, in, out,
228                                          _gpgme_decrypt_status_handler);
229   if (!err)
230       err = _gpgme_wait_one (ctx);
231   return err;
232 }