2 * Copyright (c) 2000-2002 Damien Miller. All rights reserved.
4 * Redistribution and use in source and binary forms, with or without
5 * modification, are permitted provided that the following conditions
7 * 1. Redistributions of source code must retain the above copyright
8 * notice, this list of conditions and the following disclaimer.
9 * 2. Redistributions in binary form must reproduce the above copyright
10 * notice, this list of conditions and the following disclaimer in the
11 * documentation and/or other materials provided with the distribution.
13 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
14 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
15 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
16 * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
17 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
18 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
19 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
20 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
21 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
22 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
25 /* GTK2 support by Nalin Dahyabhai <nalin@redhat.com> */
28 * This is a simple GNOME SSH passphrase grabber. To use it, set the
29 * environment variable SSH_ASKPASS to point to the location of
30 * gnome-ssh-askpass before calling "ssh-add < /dev/null".
32 * There is only two run-time options: if you set the environment variable
33 * "GNOME_SSH_ASKPASS_GRAB_SERVER=true" then gnome-ssh-askpass will grab
34 * the X server. If you set "GNOME_SSH_ASKPASS_GRAB_POINTER=true", then the
35 * pointer will be grabbed too. These may have some benefit to security if
36 * you don't trust your X server. We grab the keyboard always.
40 #define GRAB_WAIT 250 /* milliseconds */
45 * cc -Wall `pkg-config --cflags gtk+-2.0` \
46 * gnome-ssh-askpass2.c -o gnome-ssh-askpass \
47 * `pkg-config --libs gtk+-2.0`
60 report_failed_grab (const char *what
)
64 err
= gtk_message_dialog_new(NULL
, 0,
68 "A malicious client may be eavesdropping "
69 "on your session.", what
);
70 gtk_window_set_position(GTK_WINDOW(err
), GTK_WIN_POS_CENTER
);
71 gtk_label_set_line_wrap(GTK_LABEL((GTK_MESSAGE_DIALOG(err
))->label
),
74 gtk_dialog_run(GTK_DIALOG(err
));
76 gtk_widget_destroy(err
);
80 ok_dialog(GtkWidget
*entry
, gpointer dialog
)
82 g_return_if_fail(GTK_IS_DIALOG(dialog
));
83 gtk_dialog_response(GTK_DIALOG(dialog
), GTK_RESPONSE_OK
);
87 passphrase_dialog(char *message
)
90 char *passphrase
, *local
;
91 int result
, grab_tries
, grab_server
, grab_pointer
;
92 GtkWidget
*dialog
, *entry
;
95 grab_server
= (getenv("GNOME_SSH_ASKPASS_GRAB_SERVER") != NULL
);
96 grab_pointer
= (getenv("GNOME_SSH_ASKPASS_GRAB_POINTER") != NULL
);
99 dialog
= gtk_message_dialog_new(NULL
, 0,
100 GTK_MESSAGE_QUESTION
,
101 GTK_BUTTONS_OK_CANCEL
,
105 entry
= gtk_entry_new();
106 gtk_box_pack_start(GTK_BOX(GTK_DIALOG(dialog
)->vbox
), entry
, FALSE
,
108 gtk_entry_set_visibility(GTK_ENTRY(entry
), FALSE
);
109 gtk_widget_grab_focus(entry
);
110 gtk_widget_show(entry
);
112 gtk_window_set_title(GTK_WINDOW(dialog
), "OpenSSH");
113 gtk_window_set_position (GTK_WINDOW(dialog
), GTK_WIN_POS_CENTER
);
114 gtk_window_set_keep_above(GTK_WINDOW(dialog
), TRUE
);
115 gtk_label_set_line_wrap(GTK_LABEL((GTK_MESSAGE_DIALOG(dialog
))->label
),
118 /* Make <enter> close dialog */
119 gtk_dialog_set_default_response(GTK_DIALOG(dialog
), GTK_RESPONSE_OK
);
120 g_signal_connect(G_OBJECT(entry
), "activate",
121 G_CALLBACK(ok_dialog
), dialog
);
123 gtk_window_set_keep_above(GTK_WINDOW(dialog
), TRUE
);
126 gtk_widget_show_now(dialog
);
129 status
= gdk_pointer_grab(
130 (GTK_WIDGET(dialog
))->window
, TRUE
, 0, NULL
,
131 NULL
, GDK_CURRENT_TIME
);
132 if (status
== GDK_GRAB_SUCCESS
)
134 usleep(GRAB_WAIT
* 1000);
135 if (++grab_tries
> GRAB_TRIES
) {
142 status
= gdk_keyboard_grab((GTK_WIDGET(dialog
))->window
,
143 FALSE
, GDK_CURRENT_TIME
);
144 if (status
== GDK_GRAB_SUCCESS
)
146 usleep(GRAB_WAIT
* 1000);
147 if (++grab_tries
> GRAB_TRIES
) {
153 gdk_x11_grab_server();
156 result
= gtk_dialog_run(GTK_DIALOG(dialog
));
160 XUngrabServer(GDK_DISPLAY());
162 gdk_pointer_ungrab(GDK_CURRENT_TIME
);
163 gdk_keyboard_ungrab(GDK_CURRENT_TIME
);
166 /* Report passphrase if user selected OK */
167 passphrase
= g_strdup(gtk_entry_get_text(GTK_ENTRY(entry
)));
168 if (result
== GTK_RESPONSE_OK
) {
169 local
= g_locale_from_utf8(passphrase
, strlen(passphrase
),
173 memset(local
, '\0', strlen(local
));
180 /* Zero passphrase in memory */
181 memset(passphrase
, '\b', strlen(passphrase
));
182 gtk_entry_set_text(GTK_ENTRY(entry
), passphrase
);
183 memset(passphrase
, '\0', strlen(passphrase
));
186 gtk_widget_destroy(dialog
);
187 return (result
== GTK_RESPONSE_OK
? 0 : -1);
189 /* At least one grab failed - ungrab what we got, and report
190 the failure to the user. Note that XGrabServer() cannot
193 gdk_pointer_ungrab(GDK_CURRENT_TIME
);
196 XUngrabServer(GDK_DISPLAY());
197 gtk_widget_destroy(dialog
);
199 report_failed_grab(failed
);
205 main(int argc
, char **argv
)
210 gtk_init(&argc
, &argv
);
213 message
= g_strjoinv(" ", argv
+ 1);
215 message
= g_strdup("Enter your OpenSSH passphrase:");
218 setvbuf(stdout
, 0, _IONBF
, 0);
219 result
= passphrase_dialog(message
);