/[winpt]/trunk/Src/WinPT.cpp
ViewVC logotype

Annotation of /trunk/Src/WinPT.cpp

Parent Directory Parent Directory | Revision Log Revision Log


Revision 115 - (hide annotations)
Wed Dec 7 08:11:34 2005 UTC (19 years, 2 months ago) by twoaday
File size: 15102 byte(s)
2005-12-06  Timo Schulz  <ts@g10code.com>
 
        * WinPT.cpp (WinMain): Load the gettext environment soon
        as possible so all 'first start' messages are localized.
        * wptKeyManagerDlg.cpp (keymanager_dlg_proc): If a single key is
        received via the keyserver, return the gpgme key context on
        success and add it directly to the list.
        * wptKeyserverDlg.cpp (keyserver_dlg_proc): Support a param
        to store the key.
        (hkp_recv_key2): Add extra param for the fingerprint.
        (hkp_recv_key): Call hkp_recv_key2 without fingerprint.
         

1 werner 36 /* WinPT.cpp - Windows Privacy Tray (WinPT)
2     * Copyright (C) 2000-2005 Timo Schulz
3     *
4     * This file is part of WinPT.
5     *
6     * WinPT is free software; you can redistribute it and/or modify
7     * it under the terms of the GNU General Public License as published by
8     * the Free Software Foundation; either version 2 of the License, or
9     * (at your option) any later version.
10     *
11     * WinPT is distributed in the hope that it will be useful,
12     * but WITHOUT ANY WARRANTY; without even the implied warranty of
13     * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14     * GNU General Public License for more details.
15     *
16     * You should have received a copy of the GNU General Public License
17     * along with WinPT; if not, write to the Free Software Foundation,
18     * Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA
19     */
20 werner 42 #ifdef HAVE_CONFIG_H
21     #include <config.h>
22     #endif
23    
24 werner 36 #include <windows.h>
25    
26 werner 47 #include "resource.h"
27 werner 36 #include "wptTypes.h"
28     #include "wptW32API.h"
29     #include "wptVersion.h"
30     #include "wptErrors.h"
31     #include "wptGPG.h"
32     #include "wptRegistry.h"
33     #include "wptCommonCtl.h"
34     #include "wptDlgs.h"
35     #include "wptNLS.h"
36     #include "wptKeyserver.h"
37     #include "wptCard.h"
38     #include "wptFileManager.h"
39     #include "wptContext.h"
40     #include "wptCardEdit.h"
41 werner 48 #include "wptCrypto.h"
42 werner 36
43 twoaday 80 #define MIN_GPG_VER "1.4.2" /* Minimal GPG version. */
44 werner 36 #define MIN_GPGME_VER "1.2.0" /* Minimal GPGME version. */
45    
46    
47     HINSTANCE glob_hinst; /* global instance for the dialogs */
48     HWND glob_hwnd; /* global window handle for the dialogs */
49     HWND activ_hwnd;
50     LOCK mo_file;
51     int scard_support = 0;
52     int debug = 0;
53     int mobile = 0;
54     int gpg_read_only = 0;
55     char gpgver[3];
56    
57    
58     /* Load the key cache and rebuild the signature cache. */
59     static void
60     update_keycache (HWND hwnd)
61     {
62     refresh_cache_s rcs = {0};
63     rcs.kr_reload = 0;
64     rcs.kr_update = 1;
65     rcs.tr_update = 1;
66     DialogBoxParam (glob_hinst, (LPCSTR)IDD_WINPT_KEYCACHE, hwnd,
67     keycache_dlg_proc, (LPARAM)&rcs);
68     }
69    
70    
71     /* Set GPGME debug mode. If @val is 0, the debug mode is disabled. */
72     void
73     gpg_set_debug_mode (int val)
74     {
75     if (val)
76     putenv ("GPGME_DEBUG=5:gpgme.dbg");
77     else
78     putenv ("GPGME_DEBUG=");
79     }
80    
81    
82     /* Return the name of the gettext language file. */
83     static char*
84     get_gettext_lang (void)
85     {
86     char *fname;
87     fname = get_reg_entry_mo ();
88     if (!fname)
89     return NULL;
90     return fname;
91     }
92    
93    
94     /* Initialize the gettext sub system. */
95     static void
96     load_gettext (int prev_inst)
97     {
98     char *nls = NULL;
99     char *file = NULL;
100    
101     nls = get_gettext_lang ();
102     if (nls) {
103     set_gettext_file ("winpt", nls);
104     file = make_filename (nls, "winpt", "mo");
105     if (!file_exist_check (nls) && init_file_lock (&mo_file, file)) {
106     if (!prev_inst)
107     msg_box (NULL, _("Could not initizalize file lock.\n"
108     "Native Language Support"),
109     _("WinPT Error"), MB_ERR);
110     }
111     free_if_alloc (nls);
112     free_if_alloc (file);
113     }
114     }
115    
116    
117     /* check if the default key from the gpg.conf file is available in the
118     keyring. if not, bail out because encryption won't work properly then. */
119     static int
120     check_default_key (gpg_keycache_t kc)
121     {
122     gpgme_key_t key;
123     gpgme_error_t err = GPG_ERR_NO_ERROR;
124     char * defkey;
125    
126     defkey = get_gnupg_default_key ();
127     if (defkey)
128     err = gpg_keycache_find_key (kc, defkey, 0, &key);
129 twoaday 66 else
130     msg_box (NULL, _("No useable secret key found."), _("WinPT Error"), MB_ERR);
131 werner 36 free_if_alloc (defkey);
132     return err? -1 : 0;
133     }
134    
135    
136     /* Return the WinPT program file name (with full pathname). */
137     static const char *
138     get_prog_part (const char * fname, int use_cwd)
139     {
140     static char program[512];
141     char currdir[256];
142     char *cmd = NULL;
143     int j;
144    
145     memset (currdir, 0, DIM (currdir));
146     memset (program, 0, DIM (program));
147    
148     if (use_cwd) {
149     GetCurrentDirectory (DIM (currdir)-1, currdir);
150     _snprintf (program, DIM (program)-1, "%s\\%s", currdir, fname);
151     }
152     else {
153     cmd = GetCommandLine ();
154     if (cmd == NULL)
155     return NULL;
156     strncpy (currdir, cmd, sizeof (currdir)-1);
157     j = strlen (currdir);
158     while (j--) {
159     if (currdir[j] == '\\')
160     break;
161     }
162     currdir[j] = 0;
163     _snprintf (program, DIM (program)-1, "%s\\%s", currdir + 1, fname);
164     }
165     return program;
166     }
167    
168    
169     /* Check that the underlying crypto engine fullfills the minimal
170     requirements so all commands work properly. */
171     static int
172     check_crypto_engine (void)
173     {
174 twoaday 80 int ma=1, mi=4, pa=2; /* GPG 1.4.2 */
175 werner 36 int rc;
176    
177     rc = check_gnupg_engine (&ma, &mi, &pa);
178     if (rc == -1) {
179     msg_box (NULL, _("Could not read GnuPG version."),
180     _("WinPT Error"), MB_ERR);
181     return rc;
182     }
183     else if (rc) {
184     log_box (_("WinPT Error"), MB_ERR,
185     _("Sorry, you need a newer GPG version.\n"
186     "GPG version %d.%d.%d required GPG version "MIN_GPG_VER),
187     ma, mi, pa);
188     return rc;
189     }
190 twoaday 80 /* We enable smartcard support for GPG: >= 2 or >= 1.4.3 */
191     if (ma > 1 || pa >= 3)
192 werner 36 scard_support = 1;
193    
194     gpgver[0] = ma;
195     gpgver[1] = mi;
196     gpgver[2] = pa;
197     return rc;
198     }
199    
200    
201     /* Try to load the keyserver config file. If @quiet is 1
202     do not show any errors. */
203     static int
204     load_keyserver_conf (int quiet)
205     {
206     const char * t;
207     int rc;
208    
209     if (reg_prefs.kserv_conf)
210     t = reg_prefs.kserv_conf;
211     else if (!file_exist_check (get_prog_part ("keyserver.conf", 0)))
212     t = get_prog_part ("keyserver.conf", 0);
213     else
214     t = "keyserver.conf";
215     rc = kserver_load_conf (t);
216     if (rc && !quiet)
217     msg_box (NULL, winpt_strerror (rc), _("Keyserver"), MB_ERR);
218     return rc;
219     }
220    
221    
222     /* Enable the mobility mode. */
223     static void
224     enable_mobile_mode (void)
225     {
226     memset (&reg_prefs, 0, sizeof (reg_prefs));
227     reg_prefs.always_trust = 0;
228     reg_prefs.auto_backup = 0;
229     reg_prefs.cache_time = 0;
230     reg_prefs.expert = 0;
231     reg_prefs.keylist_mode = 1;
232     reg_prefs.kserv_conf = m_strdup ("keyserver.conf");
233     reg_prefs.no_zip_mmedia = 1;
234     reg_prefs.use_tmpfiles = 1;
235     reg_prefs.word_wrap = 80;
236     reg_prefs.use_viewer = 0; /* XXX */
237     }
238    
239     char* multi_gnupg_path (void);
240    
241 twoaday 41 const char * fm_get_file_type (const char *fname, int *r_type);
242    
243 werner 36 /* Main entry point. */
244     int WINAPI
245     WinMain (HINSTANCE hinst, HINSTANCE hprev, LPSTR cmdline, int showcmd)
246     {
247     WNDCLASS wc = {0, winpt_main_proc, 0, 0, hinst, 0, 0, 0, 0, PGM_NAME};
248     HACCEL accel_tab;
249 twoaday 41 int rc, ec, created = 0, nfiles = 0;
250 werner 36 int first_start = 0, start_gpgprefs = 0;
251     int winpt_inst_found = 0;
252     const char *s;
253     MSG msg;
254     HWND hwnd = NULL;
255    
256     glob_hinst = hinst;
257    
258 twoaday 87 if (cmdline && stristr (cmdline, "--stop")) {
259     hwnd = FindWindow ("WinPT", "WinPT");
260     if (hwnd != NULL)
261     PostMessage (hwnd, WM_DESTROY, 0, 0);
262     return 0;
263     }
264    
265 werner 58 #ifdef _DEBUG
266 werner 36 gpg_set_debug_mode (1);
267     debug = 1;
268 werner 58 #endif
269 werner 36
270     if (gpg_md_selftest ()) {
271     msg_box (NULL, _("Cryptographic selftest failed."),
272     _("WinPT Error"), MB_ERR);
273     return 0;
274     }
275    
276     s = gpgme_check_version (MIN_GPGME_VER);
277     if (!s || !*s) {
278     msg_box (NULL, _("A newer GPGME version is needed; at least "MIN_GPGME_VER),
279     _("WinPT Error"), MB_ERR);
280     return 0;
281     }
282    
283     CreateMutex (NULL, TRUE, PGM_NAME);
284     if (GetLastError () == ERROR_ALREADY_EXISTS)
285     winpt_inst_found = 1;
286    
287     if (cmdline && stristr (cmdline, "--mobile")) {
288     msg_box (NULL, "WARNING: mobile modus is not fully implemented yet!",
289     "WinPT", MB_INFO);
290     mobile = 1;
291     }
292    
293 twoaday 115 load_gettext (winpt_inst_found);
294 werner 36 set_default_kserver ();
295    
296     if (!mobile) {
297     regist_inst_gnupg (1);
298     regist_inst_winpt (1, &created);
299     }
300     else {
301     enable_mobile_mode ();
302     /* XXX: ask for GPG path */
303     created = 1; /* Disable registry writing */
304     }
305    
306     if (!created) {
307     memset (&reg_prefs, 0, sizeof (reg_prefs));
308     reg_prefs.use_tmpfiles = 1; /* default */
309     reg_prefs.fm.progress = 0; /* XXX: fix the bug and enable it again */
310     get_reg_winpt_prefs (&reg_prefs);
311     if (!reg_prefs.no_hotkeys)
312     hotkeys_modify ();
313 twoaday 41 gnupg_load_config ();
314 werner 36 }
315    
316     rc = gnupg_check_homedir ();
317     if (rc) {
318     log_box (_("WinPT Error"), MB_ERR,
319     _("GPG home directory is not set correctly.\n"
320     "Please check the GPG registry settings:\n%s."),
321     winpt_strerror (rc));
322 werner 48 s = get_fileopen_dlg (GetActiveWindow (),
323     _("Select GPG Public Keyring"),
324     _("GPG Keyrings (*.gpg)\0*.gpg\0\0"),
325     NULL);
326 werner 36 if (s != NULL) {
327     size_t n;
328     char * p = strrchr (s, '\\');
329     if (!p)
330     BUG (0);
331     n = p - s;
332     if (n) {
333     char * file = new char[n+1];
334     if (!file)
335     BUG (NULL);
336     memset (file, 0, n);
337     memcpy (file, s, n);
338     file[n] = '\0';
339     set_reg_entry_gpg ("HomeDir", file);
340     free_if_alloc (file);
341     gnupg_check_homedir (); /* change gpgProgram if needed */
342     }
343     }
344     else {
345     msg_box (NULL, _("GPG home directory could not be determited."),
346     _("WinPT Error"), MB_ERR);
347     goto start;
348     }
349     }
350    
351     rc = check_gnupg_prog ();
352     if (rc) {
353     if (msg_box (NULL, _("Could not find the GPG binary (gpg.exe).\n"
354     "Do you want to start the GPG preferences to "
355     "correct this problem?"), _("WinPT Error"),
356     MB_INFO|MB_YESNO) == IDYES)
357     start_gpgprefs = 1;
358     else
359     {
360     msg_box (NULL, winpt_strerror (rc), _("WinPT Error"), MB_ERR);
361     return 0;
362     }
363     }
364    
365     rc = gnupg_access_files ();
366     if (!start_gpgprefs && rc) {
367     if (rc == WPTERR_GPG_KEYRINGS || rc == WPTERR_GPG_OPT_KEYRINGS) {
368     ec = msg_box (NULL,
369     _("Could not access and/or find the public and secret keyring.\n"
370     "If this is an accident, quit the program and fix it.\n\n"
371     "Continue if you want that WinPT offers you more choices.\n"),
372     "WinPT", MB_INFO|MB_YESNO);
373     if (ec == IDYES)
374     first_start = 1;
375     }
376     if (!first_start) {
377     msg_box (NULL, winpt_strerror (rc), _("WinPT Error"), MB_ERR);
378     return 0;
379     }
380     }
381    
382     if (!first_start) {
383     rc = gpg_check_permissions (1);
384     if (rc && rc == 2)
385     gpg_read_only = 1;
386     else if (rc)
387     return 0;
388     }
389    
390     init_gnupg_table ();
391    
392     nfiles = fm_parse_command_line (cmdline);
393     if (nfiles > 0) {
394     free_gnupg_table ();
395     return 0;
396     }
397    
398     if (cmdline && stristr (cmdline, "--wipe-freespace")) {
399     dialog_box_param (glob_hinst, (LPCTSTR)IDD_WINPT_SPACE_SECDEL,
400 twoaday 73 GetDesktopWindow(), space_wipefrees_dlg_proc, 0,
401 werner 36 _("Wipe Free Space"), IDS_WINPT_SPACE_SECDEL);
402     free_gnupg_table ();
403     return 0;
404     }
405    
406     load_keyserver_conf (cmdline? 1 : 0);
407    
408     if (cmdline && (stristr (cmdline, "--keymanager")
409     || stristr (cmdline, "--cardmanager"))) {
410 twoaday 102 /* If an instance of WinPT is running, just send the command
411     to open the key manager. Otherwise start a new instance.
412     */
413     HWND tray = FindWindow ("WinPT", "WinPT");
414     if (tray != NULL) {
415     PostMessage (tray, WM_COMMAND, ID_WINPT_KEY, 0);
416     free_gnupg_table ();
417     return 0;
418     }
419 werner 36 update_keycache (GetDesktopWindow ());
420     if (stristr (cmdline, "keymanager"))
421     dialog_box_param (glob_hinst, (LPCTSTR)IDD_WINPT_KEYMISC,
422 twoaday 73 GetDesktopWindow(), keymanager_dlg_proc, 0,
423 werner 36 _("Key Manager"), IDS_WINPT_KEYMISC);
424     else {
425     gpg_card_t crd = gpg_card_load ();
426     if (crd)
427     dialog_box_param (glob_hinst, (LPCTSTR)IDD_WINPT_CARD_EDIT,
428     GetDesktopWindow(), card_edit_dlg_proc,
429     (LPARAM)crd, _("Card Manager"),
430     IDS_WINPT_CARD_EDIT);
431     gpg_card_release (crd);
432     }
433 twoaday 102 /*
434 werner 36 keycache_release (0);
435     free_gnupg_table ();
436     return 0;
437 twoaday 102 */
438 werner 36 }
439    
440     /* If we found another WinPT instance, just quit to avoid it
441     will be executed twice. */
442     if (winpt_inst_found) {
443     log_debug ("%s", "WinMain: WinPT is already running.");
444     free_gnupg_table ();
445     return 0;
446     }
447    
448     if (cmdline) {
449     if (stristr (cmdline, "--enable-debug") || stristr (cmdline, "--debug")) {
450     gpg_set_debug_mode (1);
451     winpt_debug_msg ();
452     debug = 1;
453     }
454     }
455    
456     wc.hIcon = LoadIcon (glob_hinst, MAKEINTRESOURCE (IDI_WINPT));
457     rc = RegisterClass (&wc);
458     if (rc == FALSE) {
459     msg_box (NULL, _("Could not register window class"),
460     _("WinPT Error"), MB_ERR);
461     free_gnupg_table ();
462     return 0;
463     }
464    
465     hwnd = CreateWindow (PGM_NAME,
466     PGM_NAME,
467     0, 0, 0, 0, 0,
468     NULL,
469     NULL,
470     hinst,
471     NULL);
472     if (hwnd == NULL) {
473     msg_box (NULL, _("Could not create window"), _("WinPT Error"), MB_ERR);
474     free_gnupg_table ();
475     return 0;
476     }
477     glob_hwnd = hwnd;
478     UpdateWindow (hwnd);
479    
480     if (!first_start && !start_gpgprefs) {
481     gnupg_backup_options ();
482     rc = check_crypto_engine ();
483     if (rc) {
484     DestroyWindow (hwnd);
485     free_gnupg_table ();
486     return 0;
487     }
488     }
489    
490     if (start_gpgprefs) {
491     char *ring;
492 werner 48
493 werner 36 DialogBoxParam (glob_hinst, (LPCTSTR)IDD_WINPT_GPGPREFS, hwnd,
494 twoaday 41 gpgprefs_dlg_proc, 0);
495 werner 36 ring = get_gnupg_keyring (0, !NO_STRICT);
496     if (gnupg_access_keyring (0) == -1 && get_file_size (ring) == 0)
497     first_start = 1; /* The keyring is empty! */
498     free_if_alloc (ring);
499     }
500    
501     if (first_start) {
502     struct first_start_s fs;
503     struct genkey_s c;
504     HWND h;
505     start:
506     h = GetDesktopWindow ();
507     DialogBoxParam (glob_hinst, (LPCTSTR)IDD_WINPT_GPGPREFS, h,
508 twoaday 41 gpgprefs_dlg_proc, 0);
509 werner 36 DialogBoxParam (glob_hinst, (LPCSTR)IDD_WINPT_FIRST, h,
510     first_run_dlg_proc, (LPARAM)&fs);
511     switch (fs.choice) {
512     case SETUP_KEYGEN:
513     c.interactive = 1;
514     c.first_start = 1;
515     rc = DialogBoxParam (glob_hinst, (LPCSTR)IDD_WINPT_KEYWIZARD,
516     h, keygen_wizard_dlg_proc, (LPARAM)&c);
517     if (!rc)
518     goto start;
519     break;
520    
521     case SETUP_IMPORT:
522     rc = gnupg_copy_keyrings ();
523     if (rc) {
524     msg_box (hwnd, winpt_strerror (rc), _("WinPT Error"), MB_ERR);
525     goto start;
526     }
527     break;
528    
529     case -1:
530     DestroyWindow (hwnd);
531     free_gnupg_table ();
532     return 0;
533     }
534     update_keycache (hwnd);
535     check_crypto_engine ();
536     }
537     else {
538     gpg_keycache_t c;
539     update_keycache (hwnd);
540     c = keycache_get_ctx (1);
541     if (!c || !gpg_keycache_get_size (c)) {
542     gnupg_display_error ();
543     msg_box (hwnd, _("The keycache was not initialized or is empty.\n"
544     "Please check your GPG config (keyrings, pathes...)"),
545     _("WinPT Error"), MB_ERR);
546     ec = msg_box (NULL, _("It seems that GPG is not set properly.\n"
547     "Do you want to start the GPG preferences dialog?"),
548     "WinPT", MB_INFO|MB_YESNO);
549     if (ec == IDYES) {
550     DialogBoxParam (glob_hinst, (LPCTSTR)IDD_WINPT_GPGPREFS, hwnd,
551 twoaday 41 gpgprefs_dlg_proc, 0);
552 werner 36 update_keycache (hwnd);
553     }
554     else {
555     DestroyWindow (hwnd);
556     free_gnupg_table ();
557     return 0;
558     }
559     }
560     if (check_default_key (c)) {
561     char * p = get_gnupg_default_key ();
562     log_box (_("WinPT Error"), MB_ERR,
563     _("Default key from the GPG options file could not be found.\n"
564     "Please check your gpg.conf (options) to correct this:\n\n"
565     "%s: public key not found."), p? p : "[null]");
566     free_if_alloc (p);
567     DestroyWindow (hwnd);
568     free_gnupg_table ();
569     return 0;
570     }
571     if (count_insecure_elgkeys ())
572     DialogBoxParam (glob_hinst, (LPCTSTR)IDD_WINPT_ELGWARN, glob_hwnd,
573 twoaday 41 elgamal_warn_dlg_proc, 0);
574 werner 36 }
575    
576     accel_tab = LoadAccelerators (glob_hinst, (LPCTSTR)IDR_WINPT_ACCELERATOR);
577     keyring_check_last_access (); /* init */
578     while (GetMessage (&msg, hwnd, 0, 0)) {
579     if (!TranslateAccelerator (msg.hwnd, accel_tab, &msg)) {
580     TranslateMessage (&msg);
581     DispatchMessage (&msg);
582     }
583     }
584    
585     return 0;
586     }

Properties

Name Value
svn:eol-style native

[email protected]
ViewVC Help
Powered by ViewVC 1.1.26