OLD | NEW |
| (Empty) |
1 // Copyright (c) 2010 The Chromium Authors. All rights reserved. | |
2 // Use of this source code is governed by a BSD-style license that can be | |
3 // found in the LICENSE file. | |
4 | |
5 #include "chrome/browser/ssl_client_certificate_selector.h" | |
6 | |
7 #include <gtk/gtk.h> | |
8 | |
9 #include <string> | |
10 #include <vector> | |
11 | |
12 #include "app/gtk_signal.h" | |
13 #include "app/l10n_util.h" | |
14 #include "base/i18n/time_formatting.h" | |
15 #include "base/logging.h" | |
16 #include "base/nss_util.h" | |
17 #include "base/utf_string_conversions.h" | |
18 #include "chrome/browser/certificate_viewer.h" | |
19 #include "chrome/browser/gtk/constrained_window_gtk.h" | |
20 #include "chrome/browser/gtk/gtk_util.h" | |
21 #include "chrome/browser/gtk/owned_widget_gtk.h" | |
22 #include "chrome/browser/ssl/ssl_client_auth_handler.h" | |
23 #include "chrome/browser/tab_contents/tab_contents.h" | |
24 #include "chrome/browser/ui/pk11_password_dialog.h" | |
25 #include "chrome/common/net/x509_certificate_model.h" | |
26 #include "gfx/native_widget_types.h" | |
27 #include "grit/generated_resources.h" | |
28 #include "net/base/x509_certificate.h" | |
29 | |
30 namespace { | |
31 | |
32 enum { | |
33 RESPONSE_SHOW_CERT_INFO = 1, | |
34 }; | |
35 | |
36 /////////////////////////////////////////////////////////////////////////////// | |
37 // SSLClientCertificateSelector | |
38 | |
39 class SSLClientCertificateSelector : public ConstrainedDialogDelegate { | |
40 public: | |
41 explicit SSLClientCertificateSelector( | |
42 TabContents* parent, | |
43 net::SSLCertRequestInfo* cert_request_info, | |
44 SSLClientAuthHandler* delegate); | |
45 ~SSLClientCertificateSelector(); | |
46 | |
47 void Show(); | |
48 | |
49 // ConstrainedDialogDelegate implementation: | |
50 virtual GtkWidget* GetWidgetRoot() { return root_widget_.get(); } | |
51 virtual void DeleteDelegate(); | |
52 | |
53 private: | |
54 void PopulateCerts(); | |
55 | |
56 net::X509Certificate* GetSelectedCert(); | |
57 | |
58 static std::string FormatComboBoxText( | |
59 net::X509Certificate::OSCertHandle cert, | |
60 const std::string& nickname); | |
61 static std::string FormatDetailsText( | |
62 net::X509Certificate::OSCertHandle cert); | |
63 | |
64 // Callback after unlocking certificate slot. | |
65 void Unlocked(); | |
66 | |
67 CHROMEGTK_CALLBACK_0(SSLClientCertificateSelector, void, OnComboBoxChanged); | |
68 CHROMEGTK_CALLBACK_0(SSLClientCertificateSelector, void, OnViewClicked); | |
69 CHROMEGTK_CALLBACK_0(SSLClientCertificateSelector, void, OnCancelClicked); | |
70 CHROMEGTK_CALLBACK_0(SSLClientCertificateSelector, void, OnOkClicked); | |
71 CHROMEGTK_CALLBACK_1(SSLClientCertificateSelector, void, OnPromptShown, | |
72 GtkWidget*); | |
73 | |
74 scoped_refptr<net::SSLCertRequestInfo> cert_request_info_; | |
75 | |
76 std::vector<std::string> details_strings_; | |
77 | |
78 GtkWidget* cert_combo_box_; | |
79 GtkTextBuffer* cert_details_buffer_; | |
80 | |
81 scoped_refptr<SSLClientAuthHandler> delegate_; | |
82 | |
83 OwnedWidgetGtk root_widget_; | |
84 // Hold on to the select button to focus it. | |
85 GtkWidget* select_button_; | |
86 | |
87 TabContents* parent_; | |
88 ConstrainedWindow* window_; | |
89 | |
90 DISALLOW_COPY_AND_ASSIGN(SSLClientCertificateSelector); | |
91 }; | |
92 | |
93 SSLClientCertificateSelector::SSLClientCertificateSelector( | |
94 TabContents* parent, | |
95 net::SSLCertRequestInfo* cert_request_info, | |
96 SSLClientAuthHandler* delegate) | |
97 : cert_request_info_(cert_request_info), | |
98 delegate_(delegate), | |
99 parent_(parent), | |
100 window_(NULL) { | |
101 root_widget_.Own(gtk_vbox_new(FALSE, gtk_util::kControlSpacing)); | |
102 | |
103 GtkWidget* site_vbox = gtk_vbox_new(FALSE, gtk_util::kControlSpacing); | |
104 gtk_box_pack_start(GTK_BOX(root_widget_.get()), site_vbox, | |
105 FALSE, FALSE, 0); | |
106 | |
107 GtkWidget* site_description_label = gtk_util::CreateBoldLabel( | |
108 l10n_util::GetStringUTF8(IDS_CERT_SELECTOR_SITE_DESCRIPTION_LABEL)); | |
109 gtk_box_pack_start(GTK_BOX(site_vbox), site_description_label, | |
110 FALSE, FALSE, 0); | |
111 | |
112 GtkWidget* site_label = gtk_label_new( | |
113 cert_request_info->host_and_port.c_str()); | |
114 gtk_util::LeftAlignMisc(site_label); | |
115 gtk_box_pack_start(GTK_BOX(site_vbox), site_label, FALSE, FALSE, 0); | |
116 | |
117 GtkWidget* selector_vbox = gtk_vbox_new(FALSE, gtk_util::kControlSpacing); | |
118 gtk_box_pack_start(GTK_BOX(root_widget_.get()), selector_vbox, | |
119 TRUE, TRUE, 0); | |
120 | |
121 GtkWidget* choose_description_label = gtk_util::CreateBoldLabel( | |
122 l10n_util::GetStringUTF8(IDS_CERT_SELECTOR_CHOOSE_DESCRIPTION_LABEL)); | |
123 gtk_box_pack_start(GTK_BOX(selector_vbox), choose_description_label, | |
124 FALSE, FALSE, 0); | |
125 | |
126 | |
127 cert_combo_box_ = gtk_combo_box_new_text(); | |
128 g_signal_connect(cert_combo_box_, "changed", | |
129 G_CALLBACK(OnComboBoxChangedThunk), this); | |
130 gtk_box_pack_start(GTK_BOX(selector_vbox), cert_combo_box_, | |
131 FALSE, FALSE, 0); | |
132 | |
133 GtkWidget* details_label = gtk_label_new(l10n_util::GetStringUTF8( | |
134 IDS_CERT_SELECTOR_DETAILS_DESCRIPTION_LABEL).c_str()); | |
135 gtk_util::LeftAlignMisc(details_label); | |
136 gtk_box_pack_start(GTK_BOX(selector_vbox), details_label, FALSE, FALSE, 0); | |
137 | |
138 // TODO(mattm): fix text view coloring (should have grey background). | |
139 GtkWidget* cert_details_view = gtk_text_view_new(); | |
140 gtk_text_view_set_editable(GTK_TEXT_VIEW(cert_details_view), FALSE); | |
141 gtk_text_view_set_wrap_mode(GTK_TEXT_VIEW(cert_details_view), GTK_WRAP_WORD); | |
142 cert_details_buffer_ = gtk_text_view_get_buffer( | |
143 GTK_TEXT_VIEW(cert_details_view)); | |
144 // We put the details in a frame instead of a scrolled window so that the | |
145 // entirety will be visible without requiring scrolling or expanding the | |
146 // dialog. This does however mean the dialog will grow itself if you switch | |
147 // to different cert that has longer details text. | |
148 GtkWidget* details_frame = gtk_frame_new(NULL); | |
149 gtk_frame_set_shadow_type(GTK_FRAME(details_frame), GTK_SHADOW_ETCHED_IN); | |
150 gtk_container_add(GTK_CONTAINER(details_frame), cert_details_view); | |
151 gtk_box_pack_start(GTK_BOX(selector_vbox), details_frame, TRUE, TRUE, 0); | |
152 | |
153 // And then create a set of buttons like a GtkDialog would. | |
154 GtkWidget* button_box = gtk_hbutton_box_new(); | |
155 gtk_button_box_set_layout(GTK_BUTTON_BOX(button_box), GTK_BUTTONBOX_END); | |
156 gtk_box_set_spacing(GTK_BOX(button_box), gtk_util::kControlSpacing); | |
157 gtk_box_pack_end(GTK_BOX(root_widget_.get()), button_box, FALSE, FALSE, 0); | |
158 | |
159 GtkWidget* view_button = gtk_button_new_with_mnemonic( | |
160 l10n_util::GetStringUTF8(IDS_PAGEINFO_CERT_INFO_BUTTON).c_str()); | |
161 gtk_box_pack_start(GTK_BOX(button_box), view_button, FALSE, FALSE, 0); | |
162 g_signal_connect(view_button, "clicked", | |
163 G_CALLBACK(OnViewClickedThunk), this); | |
164 | |
165 GtkWidget* cancel_button = gtk_button_new_from_stock(GTK_STOCK_CANCEL); | |
166 gtk_box_pack_end(GTK_BOX(button_box), cancel_button, FALSE, FALSE, 0); | |
167 g_signal_connect(cancel_button, "clicked", | |
168 G_CALLBACK(OnCancelClickedThunk), this); | |
169 | |
170 GtkWidget* select_button = gtk_button_new_from_stock(GTK_STOCK_OK); | |
171 gtk_box_pack_end(GTK_BOX(button_box), select_button, FALSE, FALSE, 0); | |
172 g_signal_connect(select_button, "clicked", | |
173 G_CALLBACK(OnOkClickedThunk), this); | |
174 | |
175 // When we are attached to a window, focus the select button. | |
176 select_button_ = select_button; | |
177 g_signal_connect(root_widget_.get(), "hierarchy-changed", | |
178 G_CALLBACK(OnPromptShownThunk), this); | |
179 PopulateCerts(); | |
180 | |
181 gtk_widget_show_all(root_widget_.get()); | |
182 } | |
183 | |
184 SSLClientCertificateSelector::~SSLClientCertificateSelector() { | |
185 root_widget_.Destroy(); | |
186 } | |
187 | |
188 void SSLClientCertificateSelector::Show() { | |
189 DCHECK(!window_); | |
190 window_ = parent_->CreateConstrainedDialog(this); | |
191 } | |
192 | |
193 void SSLClientCertificateSelector::DeleteDelegate() { | |
194 if (delegate_) { | |
195 // The dialog was closed by escape key. | |
196 delegate_->CertificateSelected(NULL); | |
197 } | |
198 delete this; | |
199 } | |
200 | |
201 void SSLClientCertificateSelector::PopulateCerts() { | |
202 std::vector<std::string> nicknames; | |
203 x509_certificate_model::GetNicknameStringsFromCertList( | |
204 cert_request_info_->client_certs, | |
205 l10n_util::GetStringUTF8(IDS_CERT_SELECTOR_CERT_EXPIRED), | |
206 l10n_util::GetStringUTF8(IDS_CERT_SELECTOR_CERT_NOT_YET_VALID), | |
207 &nicknames); | |
208 | |
209 DCHECK_EQ(nicknames.size(), | |
210 cert_request_info_->client_certs.size()); | |
211 | |
212 for (size_t i = 0; i < cert_request_info_->client_certs.size(); ++i) { | |
213 net::X509Certificate::OSCertHandle cert = | |
214 cert_request_info_->client_certs[i]->os_cert_handle(); | |
215 | |
216 details_strings_.push_back(FormatDetailsText(cert)); | |
217 | |
218 gtk_combo_box_append_text( | |
219 GTK_COMBO_BOX(cert_combo_box_), | |
220 FormatComboBoxText(cert, nicknames[i]).c_str()); | |
221 } | |
222 | |
223 // Auto-select the first cert. | |
224 gtk_combo_box_set_active(GTK_COMBO_BOX(cert_combo_box_), 0); | |
225 } | |
226 | |
227 net::X509Certificate* SSLClientCertificateSelector::GetSelectedCert() { | |
228 int selected = gtk_combo_box_get_active(GTK_COMBO_BOX(cert_combo_box_)); | |
229 if (selected >= 0 && | |
230 selected < static_cast<int>( | |
231 cert_request_info_->client_certs.size())) | |
232 return cert_request_info_->client_certs[selected]; | |
233 return NULL; | |
234 } | |
235 | |
236 // static | |
237 std::string SSLClientCertificateSelector::FormatComboBoxText( | |
238 net::X509Certificate::OSCertHandle cert, const std::string& nickname) { | |
239 std::string rv(nickname); | |
240 rv += " ["; | |
241 rv += x509_certificate_model::GetSerialNumberHexified(cert, ""); | |
242 rv += ']'; | |
243 return rv; | |
244 } | |
245 | |
246 // static | |
247 std::string SSLClientCertificateSelector::FormatDetailsText( | |
248 net::X509Certificate::OSCertHandle cert) { | |
249 std::string rv; | |
250 | |
251 rv += l10n_util::GetStringFUTF8( | |
252 IDS_CERT_SUBJECTNAME_FORMAT, | |
253 UTF8ToUTF16(x509_certificate_model::GetSubjectName(cert))); | |
254 | |
255 rv += "\n "; | |
256 rv += l10n_util::GetStringFUTF8( | |
257 IDS_CERT_SERIAL_NUMBER_FORMAT, | |
258 UTF8ToUTF16( | |
259 x509_certificate_model::GetSerialNumberHexified(cert, ""))); | |
260 | |
261 base::Time issued, expires; | |
262 if (x509_certificate_model::GetTimes(cert, &issued, &expires)) { | |
263 string16 issued_str = base::TimeFormatShortDateAndTime(issued); | |
264 string16 expires_str = base::TimeFormatShortDateAndTime(expires); | |
265 rv += "\n "; | |
266 rv += l10n_util::GetStringFUTF8(IDS_CERT_VALIDITY_RANGE_FORMAT, | |
267 issued_str, expires_str); | |
268 } | |
269 | |
270 std::vector<std::string> usages; | |
271 x509_certificate_model::GetUsageStrings(cert, &usages); | |
272 if (usages.size()) { | |
273 rv += "\n "; | |
274 rv += l10n_util::GetStringFUTF8(IDS_CERT_X509_EXTENDED_KEY_USAGE_FORMAT, | |
275 UTF8ToUTF16(JoinString(usages, ','))); | |
276 } | |
277 | |
278 std::string key_usage_str = x509_certificate_model::GetKeyUsageString(cert); | |
279 if (!key_usage_str.empty()) { | |
280 rv += "\n "; | |
281 rv += l10n_util::GetStringFUTF8(IDS_CERT_X509_KEY_USAGE_FORMAT, | |
282 UTF8ToUTF16(key_usage_str)); | |
283 } | |
284 | |
285 std::vector<std::string> email_addresses; | |
286 x509_certificate_model::GetEmailAddresses(cert, &email_addresses); | |
287 if (email_addresses.size()) { | |
288 rv += "\n "; | |
289 rv += l10n_util::GetStringFUTF8( | |
290 IDS_CERT_EMAIL_ADDRESSES_FORMAT, | |
291 UTF8ToUTF16(JoinString(email_addresses, ','))); | |
292 } | |
293 | |
294 rv += '\n'; | |
295 rv += l10n_util::GetStringFUTF8( | |
296 IDS_CERT_ISSUERNAME_FORMAT, | |
297 UTF8ToUTF16(x509_certificate_model::GetIssuerName(cert))); | |
298 | |
299 string16 token(UTF8ToUTF16(x509_certificate_model::GetTokenName(cert))); | |
300 if (!token.empty()) { | |
301 rv += '\n'; | |
302 rv += l10n_util::GetStringFUTF8(IDS_CERT_TOKEN_FORMAT, token); | |
303 } | |
304 | |
305 return rv; | |
306 } | |
307 | |
308 void SSLClientCertificateSelector::Unlocked() { | |
309 // TODO(mattm): refactor so we don't need to call GetSelectedCert again. | |
310 net::X509Certificate* cert = GetSelectedCert(); | |
311 delegate_->CertificateSelected(cert); | |
312 delegate_ = NULL; | |
313 DCHECK(window_); | |
314 window_->CloseConstrainedWindow(); | |
315 } | |
316 | |
317 void SSLClientCertificateSelector::OnComboBoxChanged(GtkWidget* combo_box) { | |
318 int selected = gtk_combo_box_get_active( | |
319 GTK_COMBO_BOX(cert_combo_box_)); | |
320 if (selected < 0) | |
321 return; | |
322 gtk_text_buffer_set_text(cert_details_buffer_, | |
323 details_strings_[selected].c_str(), | |
324 details_strings_[selected].size()); | |
325 } | |
326 | |
327 void SSLClientCertificateSelector::OnViewClicked(GtkWidget* button) { | |
328 net::X509Certificate* cert = GetSelectedCert(); | |
329 if (cert) { | |
330 GtkWidget* toplevel = gtk_widget_get_toplevel(root_widget_.get()); | |
331 ShowCertificateViewer(GTK_WINDOW(toplevel), cert); | |
332 } | |
333 } | |
334 | |
335 void SSLClientCertificateSelector::OnCancelClicked(GtkWidget* button) { | |
336 delegate_->CertificateSelected(NULL); | |
337 delegate_ = NULL; | |
338 DCHECK(window_); | |
339 window_->CloseConstrainedWindow(); | |
340 } | |
341 | |
342 void SSLClientCertificateSelector::OnOkClicked(GtkWidget* button) { | |
343 net::X509Certificate* cert = GetSelectedCert(); | |
344 | |
345 browser::UnlockCertSlotIfNecessary( | |
346 cert, | |
347 browser::kPK11PasswordClientAuth, | |
348 cert_request_info_->host_and_port, | |
349 NewCallback(this, &SSLClientCertificateSelector::Unlocked)); | |
350 } | |
351 | |
352 void SSLClientCertificateSelector::OnPromptShown(GtkWidget* widget, | |
353 GtkWidget* previous_toplevel) { | |
354 if (!root_widget_.get() || | |
355 !GTK_WIDGET_TOPLEVEL(gtk_widget_get_toplevel(root_widget_.get()))) | |
356 return; | |
357 GTK_WIDGET_SET_FLAGS(select_button_, GTK_CAN_DEFAULT); | |
358 gtk_widget_grab_default(select_button_); | |
359 gtk_widget_grab_focus(select_button_); | |
360 } | |
361 | |
362 } // namespace | |
363 | |
364 /////////////////////////////////////////////////////////////////////////////// | |
365 // SSLClientAuthHandler platform specific implementation: | |
366 | |
367 namespace browser { | |
368 | |
369 void ShowSSLClientCertificateSelector( | |
370 TabContents* parent, | |
371 net::SSLCertRequestInfo* cert_request_info, | |
372 SSLClientAuthHandler* delegate) { | |
373 (new SSLClientCertificateSelector(parent, | |
374 cert_request_info, | |
375 delegate))->Show(); | |
376 } | |
377 | |
378 } // namespace browser | |
OLD | NEW |