doc/
[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 static const char *
116 validity_string (GpgmeValidity val)
117 {
118   const char *s = "?";
119
120   switch (val)
121     {
122     case GPGME_VALIDITY_UNKNOWN: s = "unknown"; break;
123     case GPGME_VALIDITY_NEVER:   s = "not trusted"; break;
124     case GPGME_VALIDITY_MARGINAL:s = "marginal trusted"; break;
125     case GPGME_VALIDITY_FULL:   s = "fully trusted"; break;
126     case GPGME_VALIDITY_UNDEFINED:
127     case GPGME_VALIDITY_ULTIMATE:
128       break;
129     }
130   return s;
131 }
132
133
134 static void
135 print_sig_stat ( GpgmeCtx ctx, GpgmeSigStat status )
136 {
137     const char *s;
138     time_t created;
139     int idx;
140     GpgmeKey key;
141
142     printf ("Verification Status: %s\n", status_string (status));
143     
144     for(idx=0; (s=gpgme_get_sig_status (ctx, idx, &status, &created)); idx++ ) {
145         printf ("sig %d: created: %lu expires: %lu status: %s\n",
146                 idx, (unsigned long)created, 
147                 gpgme_get_sig_ulong_attr (ctx, idx, GPGME_ATTR_EXPIRE, 0),
148                 status_string(status) );
149         printf ("sig %d: fpr/keyid: `%s' validity: %s\n",
150                 idx, s,
151                 validity_string (gpgme_get_sig_ulong_attr
152                                  (ctx, idx, GPGME_ATTR_VALIDITY, 0)) );
153         if ( !gpgme_get_sig_key (ctx, idx, &key) ) {
154             char *p = gpgme_key_get_as_xml ( key );
155             printf ("sig %d: key object:\n%s\n", idx, p );
156             free (p);
157             gpgme_key_release (key);
158         }
159     }
160 }
161
162 int 
163 main (int argc, char **argv )
164 {
165     GpgmeCtx ctx;
166     GpgmeError err;
167     GpgmeData sig, text;
168     GpgmeSigStat status;
169     char *nota;
170     int n = 0;
171     size_t len;
172     int j;
173
174     err = gpgme_new (&ctx);
175     fail_if_err (err);
176
177   do {
178     err = gpgme_data_new_from_mem ( &text,
179                                     test_text1, strlen (test_text1), 0 );
180     fail_if_err (err);
181   #if 1
182     err = gpgme_data_new_from_mem ( &sig,
183                                     test_sig1, strlen (test_sig1), 0 );
184   #else
185     err = gpgme_data_new_from_file ( &sig, "xx1", 1 );
186   #endif
187     fail_if_err (err);
188
189     puts ("checking a valid message:\n");
190     err = gpgme_op_verify (ctx, sig, text, NULL);
191     fail_if_err (err);
192     if (!gpgme_get_sig_status (ctx, 0, &status, NULL))
193       {
194         fprintf (stderr, "%s:%d: No signature\n", __FILE__, __LINE__);
195         exit (1);
196       }
197     print_sig_stat (ctx, status);
198     if (status != GPGME_SIG_STAT_GOOD)
199       {
200         fprintf (stderr, "%s:%d: Wrong sig stat\n", __FILE__, __LINE__);
201         exit (1);
202       }
203
204     if ((nota = gpgme_get_notation (ctx)))
205         printf ("---Begin Notation---\n%s---End Notation---\n", nota );
206
207     puts ("checking a manipulated message:\n");
208     gpgme_data_release (text);
209     err = gpgme_data_new_from_mem (&text,
210                                    test_text1f, strlen (test_text1f), 0);
211     fail_if_err (err);
212     gpgme_data_rewind (sig);
213     err = gpgme_op_verify (ctx, sig, text, NULL);
214     fail_if_err (err);
215     if (!gpgme_get_sig_status (ctx, 0, &status, NULL))
216       {
217         fprintf (stderr, "%s:%d: No signature\n", __FILE__, __LINE__);
218         exit (1);
219       }
220     print_sig_stat (ctx, status);
221     if (status != GPGME_SIG_STAT_BAD)
222       {
223         fprintf (stderr, "%s:%d: Wrong sig stat\n", __FILE__, __LINE__);
224         exit (1);
225       }
226     if ((nota = gpgme_get_notation (ctx)))
227         printf ("---Begin Notation---\n%s---End Notation---\n", nota );
228
229     puts ("checking a normal signature:");
230     gpgme_data_release (sig);
231     gpgme_data_release (text);
232     err = gpgme_data_new_from_mem (&sig, test_sig2, strlen (test_sig2), 0);
233     fail_if_err (err);
234     err = gpgme_data_new (&text);
235     fail_if_err (err);
236     err = gpgme_op_verify (ctx, sig, NULL, text);
237     fail_if_err (err);
238     if (!gpgme_get_sig_status (ctx, 0, &status, NULL))
239       {
240         fprintf (stderr, "%s:%d: No signature\n", __FILE__, __LINE__);
241         exit (1);
242       }
243
244     nota = gpgme_data_release_and_get_mem (text, &len);
245     for (j = 0; j < len; j++)
246       putchar (nota[j]);
247     if (strncmp (nota, test_text1, strlen (test_text1)))
248       {
249         fprintf (stderr, "%s:%d: Wrong plaintext\n", __FILE__, __LINE__);
250         exit (1);
251       }
252    
253     print_sig_stat (ctx, status);
254     if (status != GPGME_SIG_STAT_GOOD)
255       {
256         fprintf (stderr, "%s:%d: Wrong sig stat\n", __FILE__, __LINE__);
257         exit (1);
258       }
259
260     if ((nota = gpgme_get_notation (ctx)))
261       printf ("---Begin Notation---\n%s---End Notation---\n", nota);
262
263     gpgme_data_release (sig);
264  
265 } while ( argc > 1 && !strcmp( argv[1], "--loop" ) && ++n < 20 );
266       gpgme_release (ctx);
267     
268     return 0;
269 }