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

Annotation of /trunk/Src/WinPT.cpp

Parent Directory Parent Directory | Revision Log Revision Log


Revision 102 - (hide annotations)
Tue Nov 29 08:56:21 2005 UTC (19 years, 3 months ago) by twoaday
File size: 15098 byte(s)
2005-11-29  Timo Schulz  <ts@g10code.com>
 
        * WinPT.cpp (WinMain): Modify --keymanager. If an instance
        is running, just send the command to open the manager.
         


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     set_default_kserver ();
294    
295     if (!mobile) {
296     regist_inst_gnupg (1);
297     regist_inst_winpt (1, &created);
298     }
299     else {
300     enable_mobile_mode ();
301     /* XXX: ask for GPG path */
302     created = 1; /* Disable registry writing */
303     }
304    
305     if (!created) {
306     memset (&reg_prefs, 0, sizeof (reg_prefs));
307     reg_prefs.use_tmpfiles = 1; /* default */
308     reg_prefs.fm.progress = 0; /* XXX: fix the bug and enable it again */
309     get_reg_winpt_prefs (&reg_prefs);
310     if (!reg_prefs.no_hotkeys)
311     hotkeys_modify ();
312 twoaday 41 gnupg_load_config ();
313 werner 36 }
314    
315     rc = gnupg_check_homedir ();
316     if (rc) {
317     log_box (_("WinPT Error"), MB_ERR,
318     _("GPG home directory is not set correctly.\n"
319     "Please check the GPG registry settings:\n%s."),
320     winpt_strerror (rc));
321 werner 48 s = get_fileopen_dlg (GetActiveWindow (),
322     _("Select GPG Public Keyring"),
323     _("GPG Keyrings (*.gpg)\0*.gpg\0\0"),
324     NULL);
325 werner 36 if (s != NULL) {
326     size_t n;
327     char * p = strrchr (s, '\\');
328     if (!p)
329     BUG (0);
330     n = p - s;
331     if (n) {
332     char * file = new char[n+1];
333     if (!file)
334     BUG (NULL);
335     memset (file, 0, n);
336     memcpy (file, s, n);
337     file[n] = '\0';
338     set_reg_entry_gpg ("HomeDir", file);
339     free_if_alloc (file);
340     gnupg_check_homedir (); /* change gpgProgram if needed */
341     }
342     }
343     else {
344     msg_box (NULL, _("GPG home directory could not be determited."),
345     _("WinPT Error"), MB_ERR);
346     goto start;
347     }
348     }
349    
350     rc = check_gnupg_prog ();
351     if (rc) {
352     if (msg_box (NULL, _("Could not find the GPG binary (gpg.exe).\n"
353     "Do you want to start the GPG preferences to "
354     "correct this problem?"), _("WinPT Error"),
355     MB_INFO|MB_YESNO) == IDYES)
356     start_gpgprefs = 1;
357     else
358     {
359     msg_box (NULL, winpt_strerror (rc), _("WinPT Error"), MB_ERR);
360     return 0;
361     }
362     }
363    
364     rc = gnupg_access_files ();
365     if (!start_gpgprefs && rc) {
366     if (rc == WPTERR_GPG_KEYRINGS || rc == WPTERR_GPG_OPT_KEYRINGS) {
367     ec = msg_box (NULL,
368     _("Could not access and/or find the public and secret keyring.\n"
369     "If this is an accident, quit the program and fix it.\n\n"
370     "Continue if you want that WinPT offers you more choices.\n"),
371     "WinPT", MB_INFO|MB_YESNO);
372     if (ec == IDYES)
373     first_start = 1;
374     }
375     if (!first_start) {
376     msg_box (NULL, winpt_strerror (rc), _("WinPT Error"), MB_ERR);
377     return 0;
378     }
379     }
380    
381     if (!first_start) {
382     rc = gpg_check_permissions (1);
383     if (rc && rc == 2)
384     gpg_read_only = 1;
385     else if (rc)
386     return 0;
387     }
388    
389     load_gettext (winpt_inst_found);
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