Add offline mode support for CMS keylisting
[gpgme.git] / tests / run-keylist.c
1 /* run-keylist.c  - Helper to show a key listing.
2    Copyright (C) 2008, 2009 g10 Code GmbH
3
4    This file is part of GPGME.
5
6    GPGME is free software; you can redistribute it and/or modify it
7    under the terms of the GNU Lesser General Public License as
8    published by the Free Software Foundation; either version 2.1 of
9    the License, or (at your option) any later version.
10
11    GPGME is distributed in the hope that it will be useful, but
12    WITHOUT ANY WARRANTY; without even the implied warranty of
13    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
14    Lesser General Public License for more details.
15
16    You should have received a copy of the GNU Lesser General Public
17    License along with this program; if not, see <http://www.gnu.org/licenses/>.
18 */
19
20 /* We need to include config.h so that we know whether we are building
21    with large file system (LFS) support. */
22 #ifdef HAVE_CONFIG_H
23 #include <config.h>
24 #endif
25
26 #include <stdlib.h>
27 #include <stdio.h>
28 #include <string.h>
29
30 #include <gpgme.h>
31
32 #define PGM "run-keylist"
33
34 #include "run-support.h"
35
36
37 static int verbose;
38
39
40 static int
41 show_usage (int ex)
42 {
43   fputs ("usage: " PGM " [options] [USERID]\n\n"
44          "Options:\n"
45          "  --verbose        run in verbose mode\n"
46          "  --openpgp        use the OpenPGP protocol (default)\n"
47          "  --cms            use the CMS protocol\n"
48          "  --secret         list only secret keys\n"
49          "  --local          use GPGME_KEYLIST_MODE_LOCAL\n"
50          "  --extern         use GPGME_KEYLIST_MODE_EXTERN\n"
51          "  --sigs           use GPGME_KEYLIST_MODE_SIGS\n"
52          "  --sig-notations  use GPGME_KEYLIST_MODE_SIG_NOTATIONS\n"
53          "  --ephemeral      use GPGME_KEYLIST_MODE_EPHEMERAL\n"
54          "  --validate       use GPGME_KEYLIST_MODE_VALIDATE\n"
55          "  --import         import all keys\n"
56          "  --offline        use offline mode\n"
57          , stderr);
58   exit (ex);
59 }
60
61
62 int
63 main (int argc, char **argv)
64 {
65   int last_argc = -1;
66   gpgme_error_t err;
67   gpgme_ctx_t ctx;
68   gpgme_keylist_mode_t mode = 0;
69   gpgme_key_t key;
70   gpgme_keylist_result_t result;
71   int import = 0;
72   gpgme_key_t keyarray[100];
73   int keyidx = 0;
74   gpgme_protocol_t protocol = GPGME_PROTOCOL_OpenPGP;
75   int only_secret = 0;
76   int offline = 0;
77
78   if (argc)
79     { argc--; argv++; }
80
81   while (argc && last_argc != argc )
82     {
83       last_argc = argc;
84       if (!strcmp (*argv, "--"))
85         {
86           argc--; argv++;
87           break;
88         }
89       else if (!strcmp (*argv, "--help"))
90         show_usage (0);
91       else if (!strcmp (*argv, "--verbose"))
92         {
93           verbose = 1;
94           argc--; argv++;
95         }
96       else if (!strcmp (*argv, "--openpgp"))
97         {
98           protocol = GPGME_PROTOCOL_OpenPGP;
99           argc--; argv++;
100         }
101       else if (!strcmp (*argv, "--cms"))
102         {
103           protocol = GPGME_PROTOCOL_CMS;
104           argc--; argv++;
105         }
106       else if (!strcmp (*argv, "--secret"))
107         {
108           only_secret = 1;
109           argc--; argv++;
110         }
111       else if (!strcmp (*argv, "--local"))
112         {
113           mode |= GPGME_KEYLIST_MODE_LOCAL;
114           argc--; argv++;
115         }
116       else if (!strcmp (*argv, "--extern"))
117         {
118           mode |= GPGME_KEYLIST_MODE_EXTERN;
119           argc--; argv++;
120         }
121       else if (!strcmp (*argv, "--sigs"))
122         {
123           mode |= GPGME_KEYLIST_MODE_SIGS;
124           argc--; argv++;
125         }
126       else if (!strcmp (*argv, "--sig-notations"))
127         {
128           mode |= GPGME_KEYLIST_MODE_SIG_NOTATIONS;
129           argc--; argv++;
130         }
131       else if (!strcmp (*argv, "--ephemeral"))
132         {
133           mode |= GPGME_KEYLIST_MODE_EPHEMERAL;
134           argc--; argv++;
135         }
136       else if (!strcmp (*argv, "--validate"))
137         {
138           mode |= GPGME_KEYLIST_MODE_VALIDATE;
139           argc--; argv++;
140         }
141       else if (!strcmp (*argv, "--import"))
142         {
143           import = 1;
144           argc--; argv++;
145         }
146       else if (!strcmp (*argv, "--offline"))
147         {
148           offline = 1;
149           argc--; argv++;
150         }
151       else if (!strncmp (*argv, "--", 2))
152         show_usage (1);
153
154     }
155
156   if (argc > 1)
157     show_usage (1);
158
159   init_gpgme (protocol);
160
161   err = gpgme_new (&ctx);
162   fail_if_err (err);
163   gpgme_set_protocol (ctx, protocol);
164
165   gpgme_set_keylist_mode (ctx, mode);
166
167   gpgme_set_offline (ctx, offline);
168
169   err = gpgme_op_keylist_start (ctx, argc? argv[0]:NULL, only_secret);
170   fail_if_err (err);
171
172   while (!(err = gpgme_op_keylist_next (ctx, &key)))
173     {
174       gpgme_user_id_t uid;
175       int nuids;
176
177
178       printf ("keyid   : %s\n", key->subkeys?nonnull (key->subkeys->keyid):"?");
179       printf ("fpr     : %s\n", key->subkeys?nonnull (key->subkeys->fpr):"?");
180       printf ("caps    : %s%s%s%s\n",
181               key->can_encrypt? "e":"",
182               key->can_sign? "s":"",
183               key->can_certify? "c":"",
184               key->can_authenticate? "a":"");
185       printf ("flags   :%s%s%s%s%s%s\n",
186               key->secret? " secret":"",
187               key->revoked? " revoked":"",
188               key->expired? " expired":"",
189               key->disabled? " disabled":"",
190               key->invalid? " invalid":"",
191               key->is_qualified? " qualifid":"");
192       for (nuids=0, uid=key->uids; uid; uid = uid->next, nuids++)
193         {
194           printf ("userid %d: %s\n", nuids, nonnull(uid->uid));
195           printf ("valid  %d: %s\n", nuids,
196                   uid->validity == GPGME_VALIDITY_UNKNOWN? "unknown":
197                   uid->validity == GPGME_VALIDITY_UNDEFINED? "undefined":
198                   uid->validity == GPGME_VALIDITY_NEVER? "never":
199                   uid->validity == GPGME_VALIDITY_MARGINAL? "marginal":
200                   uid->validity == GPGME_VALIDITY_FULL? "full":
201                   uid->validity == GPGME_VALIDITY_ULTIMATE? "ultimate": "[?]");
202         }
203
204       putchar ('\n');
205
206       if (import)
207         {
208           if (keyidx < DIM (keyarray)-1)
209             keyarray[keyidx++] = key;
210           else
211             {
212               fprintf (stderr, PGM": too many keys in import mode"
213                        "- skipping this key\n");
214               gpgme_key_unref (key);
215             }
216         }
217       else
218         gpgme_key_unref (key);
219     }
220   if (gpgme_err_code (err) != GPG_ERR_EOF)
221     fail_if_err (err);
222   err = gpgme_op_keylist_end (ctx);
223   fail_if_err (err);
224   keyarray[keyidx] = NULL;
225
226   result = gpgme_op_keylist_result (ctx);
227   if (result->truncated)
228     {
229       fprintf (stderr, PGM ": key listing unexpectedly truncated\n");
230       exit (1);
231     }
232
233   if (import)
234     {
235       gpgme_import_result_t impres;
236
237       err = gpgme_op_import_keys (ctx, keyarray);
238       fail_if_err (err);
239       impres = gpgme_op_import_result (ctx);
240       if (!impres)
241         {
242           fprintf (stderr, PGM ": no import result returned\n");
243           exit (1);
244         }
245       print_import_result (impres);
246     }
247
248   for (keyidx=0; keyarray[keyidx]; keyidx++)
249     gpgme_key_unref (keyarray[keyidx]);
250
251   gpgme_release (ctx);
252   return 0;
253 }