7b5bc36e8706e1863bc96fcc7db0a8404e09502f
[gpgme.git] / tests / gpg / t-verify.c
1 /* t-verify.c  - regression test
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 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 #include <stdio.h>
22 #include <stdlib.h>
23 #include <string.h>
24 #include <assert.h>
25
26 #include <gpgme.h>
27
28 static const char test_text1[] = "Just GNU it!\n";
29 static const char test_text1f[]= "Just GNU it?\n";
30 static const char test_sig1[] =
31 #if 0
32 "-----BEGIN PGP SIGNATURE-----\n"
33 "\n"
34 "iEYEABECAAYFAjoKgjIACgkQLXJ8x2hpdzQMSwCeO/xUrhysZ7zJKPf/FyXA//u1\n"
35 "ZgIAn0204PBR7yxSdQx6CFxugstNqmRv\n"
36 "=yku6\n"
37 "-----END PGP SIGNATURE-----\n"
38 #elif 0
39 "-----BEGIN PGP SIGNATURE-----\n"
40 "Version: GnuPG v1.0.4-2 (GNU/Linux)\n"
41 "Comment: For info see http://www.gnupg.org\n"
42 "\n"
43 "iJcEABECAFcFAjoS8/E1FIAAAAAACAAkZm9vYmFyLjF0aGlzIGlzIGEgbm90YXRp\n"
44 "b24gZGF0YSB3aXRoIDIgbGluZXMaGmh0dHA6Ly93d3cuZ3Uub3JnL3BvbGljeS8A\n"
45 "CgkQLXJ8x2hpdzQLyQCbBW/fgU8ZeWSlWPM1F8umHX17bAAAoIfSNDSp5zM85XcG\n"
46 "iwxMrf+u8v4r\n"
47 "=88Zo\n"
48 "-----END PGP SIGNATURE-----\n"
49 #elif 1
50 "-----BEGIN PGP SIGNATURE-----\n"
51 "\n"
52 "iN0EABECAJ0FAjoS+i9FFIAAAAAAAwA5YmFyw7bDpMO8w58gZGFzIHdhcmVuIFVt\n"
53 "bGF1dGUgdW5kIGpldHp0IGVpbiBwcm96ZW50JS1aZWljaGVuNRSAAAAAAAgAJGZv\n"
54 "b2Jhci4xdGhpcyBpcyBhIG5vdGF0aW9uIGRhdGEgd2l0aCAyIGxpbmVzGhpodHRw\n"
55 "Oi8vd3d3Lmd1Lm9yZy9wb2xpY3kvAAoJEC1yfMdoaXc0JBIAoIiLlUsvpMDOyGEc\n"
56 "dADGKXF/Hcb+AKCJWPphZCphduxSvrzH0hgzHdeQaA==\n"
57 "=nts1\n"
58 "-----END PGP SIGNATURE-----\n"
59 #endif
60 ;
61 static const char test_sig2[] =
62 "-----BEGIN PGP MESSAGE-----\n"
63 "\n"
64 "owGbwMvMwCSoW1RzPCOz3IRxjXQSR0lqcYleSUWJTZOvjVdpcYmCu1+oQmaJIleH\n"
65 "GwuDIBMDGysTSIqBi1MApi+nlGGuwDeHao53HBr+FoVGP3xX+kvuu9fCMJvl6IOf\n"
66 "y1kvP4y+8D5a11ang0udywsA\n"
67 "=Crq6\n"
68 "-----END PGP MESSAGE-----\n";
69
70
71 #define fail_if_err(a) do { if(a) {                                       \
72                                fprintf (stderr, "%s:%d: GpgmeError %s\n", \
73                                 __FILE__, __LINE__, gpgme_strerror(a));   \
74                                 exit (1); }                               \
75                              } while(0)
76
77
78 static const char *
79 status_string (GpgmeSigStat status)
80 {
81     const char *s = "?";
82
83     switch ( status ) {
84       case GPGME_SIG_STAT_NONE:
85         s = "None";
86         break;
87       case GPGME_SIG_STAT_NOSIG:
88         s = "No Signature";
89         break;
90       case GPGME_SIG_STAT_GOOD:
91         s = "Good";
92         break;
93       case GPGME_SIG_STAT_GOOD_EXP:
94         s = "Good but expired";
95         break;
96       case GPGME_SIG_STAT_GOOD_EXPKEY:
97         s = "Good but key exipired";
98         break;
99       case GPGME_SIG_STAT_BAD:
100         s = "Bad";
101         break;
102       case GPGME_SIG_STAT_NOKEY:
103         s = "No Key";
104         break;
105       case GPGME_SIG_STAT_ERROR:
106         s = "Error";
107         break;
108       case GPGME_SIG_STAT_DIFF:
109         s = "More than one signature";
110         break;
111     }
112     return s;
113 }
114
115
116 static const char *
117 validity_string (GpgmeValidity val)
118 {
119   const char *s = "?";
120
121   switch (val)
122     {
123     case GPGME_VALIDITY_UNKNOWN: s = "unknown"; break;
124     case GPGME_VALIDITY_NEVER:   s = "not trusted"; break;
125     case GPGME_VALIDITY_MARGINAL:s = "marginal trusted"; break;
126     case GPGME_VALIDITY_FULL:   s = "fully trusted"; break;
127     case GPGME_VALIDITY_UNDEFINED:
128     case GPGME_VALIDITY_ULTIMATE:
129       break;
130     }
131   return s;
132 }
133
134
135 static void
136 print_sig_stat (GpgmeCtx ctx, GpgmeSigStat status)
137 {
138   const char *s;
139   time_t created;
140   int idx;
141   GpgmeKey key;
142
143   printf ("Verification Status: %s\n", status_string (status));
144     
145   for (idx = 0; (s = gpgme_get_sig_status (ctx, idx, &status, &created)); idx++)
146     {
147       printf ("sig %d: created: %lu expires: %lu status: %s\n",
148               idx, (unsigned long) created, 
149               gpgme_get_sig_ulong_attr (ctx, idx, GPGME_ATTR_EXPIRE, 0),
150               status_string (status));
151       printf ("sig %d: fpr/keyid: `%s' validity: %s\n",
152               idx, s,
153               validity_string (gpgme_get_sig_ulong_attr
154                                (ctx, idx, GPGME_ATTR_VALIDITY, 0)));
155       if (!gpgme_get_sig_key (ctx, idx, &key))
156         {
157           char *p = gpgme_key_get_as_xml (key);
158           printf ("sig %d: key object:\n%s\n", idx, p);
159           free (p);
160           gpgme_key_release (key);
161         }
162     }
163 }
164
165 int 
166 main (int argc, char *argv[])
167 {
168   GpgmeCtx ctx;
169   GpgmeError err;
170   GpgmeData sig, text;
171   GpgmeSigStat status;
172   GpgmeVerifyResult result;
173   GpgmeSigNotation notation;
174   char *nota;
175   int n = 0;
176   size_t len;
177   int j;
178
179   err = gpgme_new (&ctx);
180   fail_if_err (err);
181
182   do
183     {
184       err = gpgme_data_new_from_mem (&text,
185                                      test_text1, strlen (test_text1), 0);
186       fail_if_err (err);
187 #if 1
188       err = gpgme_data_new_from_mem (&sig,
189                                      test_sig1, strlen (test_sig1), 0);
190 #else
191       err = gpgme_data_new_from_file (&sig, "xx1", 1);
192 #endif
193       fail_if_err (err);
194
195       puts ("checking a valid message:\n");
196       err = gpgme_op_verify (ctx, sig, text, NULL);
197       fail_if_err (err);
198       if (!gpgme_get_sig_status (ctx, 0, &status, NULL))
199         {
200           fprintf (stderr, "%s:%d: No signature\n", __FILE__, __LINE__);
201           exit (1);
202         }
203       print_sig_stat (ctx, status);
204       if (status != GPGME_SIG_STAT_GOOD)
205         {
206           fprintf (stderr, "%s:%d: Wrong sig stat\n", __FILE__, __LINE__);
207           exit (1);
208         }
209
210       result = gpgme_op_verify_result (ctx);
211       notation = result->signatures->notations;
212       if (notation)
213         {
214           printf ("---Begin Notation---\n");
215           while (notation)
216             {
217               if (notation->name)
218                 printf ("%s: %s\n", notation->name, notation->value);
219               else
220                 printf ("Policy URL: %s\n", notation->value);
221               notation = notation->next;
222             }
223           printf ("---End Notation---\n");
224         }
225       
226       puts ("checking a manipulated message:\n");
227       gpgme_data_release (text);
228       err = gpgme_data_new_from_mem (&text,
229                                      test_text1f, strlen (test_text1f), 0);
230       fail_if_err (err);
231       gpgme_data_rewind (sig);
232       err = gpgme_op_verify (ctx, sig, text, NULL);
233       fail_if_err (err);
234       if (!gpgme_get_sig_status (ctx, 0, &status, NULL))
235         {
236           fprintf (stderr, "%s:%d: No signature\n", __FILE__, __LINE__);
237           exit (1);
238         }
239       print_sig_stat (ctx, status);
240       if (status != GPGME_SIG_STAT_BAD)
241         {
242           fprintf (stderr, "%s:%d: Wrong sig stat\n", __FILE__, __LINE__);
243           exit (1);
244         }
245       result = gpgme_op_verify_result (ctx);
246       notation = result->signatures->notations;
247       if (notation)
248         {
249           printf ("---Begin Notation---\n");
250           while (notation)
251             {
252               if (notation->name)
253                 printf ("%s: %s\n", notation->name, notation->value);
254               else
255                 printf ("Policy URL: %s\n", notation->value);
256               notation = notation->next;
257             }
258           printf ("---End Notation---\n");
259         }
260       
261       puts ("checking a normal signature:");
262       gpgme_data_release (sig);
263       gpgme_data_release (text);
264       err = gpgme_data_new_from_mem (&sig, test_sig2, strlen (test_sig2), 0);
265       fail_if_err (err);
266       err = gpgme_data_new (&text);
267       fail_if_err (err);
268       err = gpgme_op_verify (ctx, sig, NULL, text);
269       fail_if_err (err);
270       if (!gpgme_get_sig_status (ctx, 0, &status, NULL))
271         {
272           fprintf (stderr, "%s:%d: No signature\n", __FILE__, __LINE__);
273           exit (1);
274         }
275       
276       nota = gpgme_data_release_and_get_mem (text, &len);
277       for (j = 0; j < len; j++)
278       putchar (nota[j]);
279       if (strncmp (nota, test_text1, strlen (test_text1)))
280         {
281           fprintf (stderr, "%s:%d: Wrong plaintext\n", __FILE__, __LINE__);
282           exit (1);
283         }
284       
285       print_sig_stat (ctx, status);
286       if (status != GPGME_SIG_STAT_GOOD)
287         {
288           fprintf (stderr, "%s:%d: Wrong sig stat\n", __FILE__, __LINE__);
289           exit (1);
290         }
291       result = gpgme_op_verify_result (ctx);
292       notation = result->signatures->notations;
293       if (notation)
294         {
295           printf ("---Begin Notation---\n");
296           while (notation)
297             {
298               if (notation->name)
299                 printf ("%s: %s\n", notation->name, notation->value);
300               else
301                 printf ("Policy URL: %s\n", notation->value);
302               notation = notation->next;
303             }
304           printf ("---End Notation---\n");
305         }
306       
307       gpgme_data_release (sig);      
308     }
309   while (argc > 1 && !strcmp (argv[1], "--loop") && ++n < 20);
310
311   gpgme_release (ctx);
312   return 0;
313 }