// Copyright (c) 2010 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "chrome/renderer/localized_error.h" #include "app/l10n_util.h" #include "base/i18n/rtl.h" #include "base/logging.h" #include "base/string16.h" #include "base/string_number_conversions.h" #include "base/utf_string_conversions.h" #include "base/values.h" #include "chrome/renderer/extensions/extension_renderer_info.h" #include "googleurl/src/gurl.h" #include "grit/generated_resources.h" #include "net/base/escape.h" #include "net/base/net_errors.h" #include "third_party/WebKit/WebKit/chromium/public/WebURLError.h" #include "webkit/glue/webkit_glue.h" using WebKit::WebURLError; namespace { static const char* kRedirectLoopLearnMoreUrl = "http://www.google.com/support/chrome/bin/answer.py?answer=95626"; enum NAV_SUGGESTIONS { SUGGEST_NONE = 0, SUGGEST_RELOAD = 1 << 0, SUGGEST_HOSTNAME = 1 << 1, SUGGEST_LEARNMORE = 1 << 2, }; struct LocalizedErrorMap { int error_code; unsigned int title_resource_id; unsigned int heading_resource_id; unsigned int summary_resource_id; unsigned int details_resource_id; int suggestions; // Bitmap of SUGGEST_* values. }; const LocalizedErrorMap net_error_options[] = { {net::ERR_TIMED_OUT, IDS_ERRORPAGES_TITLE_NOT_AVAILABLE, IDS_ERRORPAGES_HEADING_NOT_AVAILABLE, IDS_ERRORPAGES_SUMMARY_NOT_AVAILABLE, IDS_ERRORPAGES_DETAILS_TIMED_OUT, SUGGEST_RELOAD, }, {net::ERR_CONNECTION_TIMED_OUT, IDS_ERRORPAGES_TITLE_NOT_AVAILABLE, IDS_ERRORPAGES_HEADING_NOT_AVAILABLE, IDS_ERRORPAGES_SUMMARY_NOT_AVAILABLE, IDS_ERRORPAGES_DETAILS_TIMED_OUT, SUGGEST_RELOAD, }, {net::ERR_CONNECTION_FAILED, IDS_ERRORPAGES_TITLE_NOT_AVAILABLE, IDS_ERRORPAGES_HEADING_NOT_AVAILABLE, IDS_ERRORPAGES_SUMMARY_NOT_AVAILABLE, IDS_ERRORPAGES_DETAILS_CONNECT_FAILED, SUGGEST_RELOAD, }, {net::ERR_NAME_NOT_RESOLVED, IDS_ERRORPAGES_TITLE_NOT_AVAILABLE, IDS_ERRORPAGES_HEADING_NOT_AVAILABLE, IDS_ERRORPAGES_SUMMARY_NOT_AVAILABLE, IDS_ERRORPAGES_DETAILS_NAME_NOT_RESOLVED, SUGGEST_RELOAD, }, {net::ERR_INTERNET_DISCONNECTED, IDS_ERRORPAGES_TITLE_NOT_AVAILABLE, IDS_ERRORPAGES_HEADING_NOT_AVAILABLE, IDS_ERRORPAGES_SUMMARY_NOT_AVAILABLE, IDS_ERRORPAGES_DETAILS_DISCONNECTED, SUGGEST_RELOAD, }, {net::ERR_FILE_NOT_FOUND, IDS_ERRORPAGES_TITLE_NOT_FOUND, IDS_ERRORPAGES_HEADING_NOT_FOUND, IDS_ERRORPAGES_SUMMARY_NOT_FOUND, IDS_ERRORPAGES_DETAILS_FILE_NOT_FOUND, SUGGEST_NONE, }, {net::ERR_TOO_MANY_REDIRECTS, IDS_ERRORPAGES_TITLE_LOAD_FAILED, IDS_ERRORPAGES_HEADING_TOO_MANY_REDIRECTS, IDS_ERRORPAGES_SUMMARY_TOO_MANY_REDIRECTS, IDS_ERRORPAGES_DETAILS_TOO_MANY_REDIRECTS, SUGGEST_RELOAD | SUGGEST_LEARNMORE, }, {net::ERR_SSL_PROTOCOL_ERROR, IDS_ERRORPAGES_TITLE_LOAD_FAILED, IDS_ERRORPAGES_HEADING_SSL_PROTOCOL_ERROR, IDS_ERRORPAGES_SUMMARY_SSL_PROTOCOL_ERROR, IDS_ERRORPAGES_DETAILS_SSL_PROTOCOL_ERROR, SUGGEST_NONE, }, {net::ERR_SSL_UNSAFE_NEGOTIATION, IDS_ERRORPAGES_TITLE_LOAD_FAILED, IDS_ERRORPAGES_HEADING_SSL_PROTOCOL_ERROR, IDS_ERRORPAGES_SUMMARY_SSL_PROTOCOL_ERROR, IDS_ERRORPAGES_DETAILS_SSL_UNSAFE_NEGOTIATION, SUGGEST_NONE, }, {net::ERR_BAD_SSL_CLIENT_AUTH_CERT, IDS_ERRORPAGES_TITLE_LOAD_FAILED, IDS_ERRORPAGES_HEADING_BAD_SSL_CLIENT_AUTH_CERT, IDS_ERRORPAGES_SUMMARY_BAD_SSL_CLIENT_AUTH_CERT, IDS_ERRORPAGES_DETAILS_BAD_SSL_CLIENT_AUTH_CERT, SUGGEST_NONE, }, }; const LocalizedErrorMap http_error_options[] = { {403, IDS_ERRORPAGES_TITLE_ACCESS_DENIED, IDS_ERRORPAGES_HEADING_ACCESS_DENIED, IDS_ERRORPAGES_SUMMARY_FORBIDDEN, IDS_ERRORPAGES_DETAILS_FORBIDDEN, SUGGEST_NONE, }, {410, IDS_ERRORPAGES_TITLE_NOT_FOUND, IDS_ERRORPAGES_HEADING_NOT_FOUND, IDS_ERRORPAGES_SUMMARY_GONE, IDS_ERRORPAGES_DETAILS_GONE, SUGGEST_NONE, }, {500, IDS_ERRORPAGES_TITLE_LOAD_FAILED, IDS_ERRORPAGES_HEADING_HTTP_SERVER_ERROR, IDS_ERRORPAGES_SUMMARY_INTERNAL_SERVER_ERROR, IDS_ERRORPAGES_DETAILS_INTERNAL_SERVER_ERROR, SUGGEST_RELOAD, }, {501, IDS_ERRORPAGES_TITLE_LOAD_FAILED, IDS_ERRORPAGES_HEADING_HTTP_SERVER_ERROR, IDS_ERRORPAGES_SUMMARY_WEBSITE_CANNOT_HANDLE, IDS_ERRORPAGES_DETAILS_NOT_IMPLEMENTED, SUGGEST_NONE, }, {502, IDS_ERRORPAGES_TITLE_LOAD_FAILED, IDS_ERRORPAGES_HEADING_HTTP_SERVER_ERROR, IDS_ERRORPAGES_SUMMARY_BAD_GATEWAY, IDS_ERRORPAGES_DETAILS_BAD_GATEWAY, SUGGEST_RELOAD, }, {503, IDS_ERRORPAGES_TITLE_LOAD_FAILED, IDS_ERRORPAGES_HEADING_HTTP_SERVER_ERROR, IDS_ERRORPAGES_SUMMARY_SERVICE_UNAVAILABLE, IDS_ERRORPAGES_DETAILS_SERVICE_UNAVAILABLE, SUGGEST_RELOAD, }, {504, IDS_ERRORPAGES_TITLE_LOAD_FAILED, IDS_ERRORPAGES_HEADING_HTTP_SERVER_ERROR, IDS_ERRORPAGES_SUMMARY_GATEWAY_TIMEOUT, IDS_ERRORPAGES_DETAILS_GATEWAY_TIMEOUT, SUGGEST_RELOAD, }, {505, IDS_ERRORPAGES_TITLE_LOAD_FAILED, IDS_ERRORPAGES_HEADING_HTTP_SERVER_ERROR, IDS_ERRORPAGES_SUMMARY_WEBSITE_CANNOT_HANDLE, IDS_ERRORPAGES_DETAILS_HTTP_VERSION_NOT_SUPPORTED, SUGGEST_NONE, }, }; const char* HttpErrorToString(int status_code) { switch (status_code) { case 403: return "Forbidden"; case 410: return "Gone"; case 500: return "Internal Server Error"; case 501: return "Not Implemented"; case 502: return "Bad Gateway"; case 503: return "Service Unavailable"; case 504: return "Gateway Timeout"; case 505: return "HTTP Version Not Supported"; default: return ""; } } string16 GetErrorDetailsString(const std::string& error_domain, int error_code, const string16& details) { int error_page_template; const char* error_string; if (error_domain == net::kErrorDomain) { error_page_template = IDS_ERRORPAGES_DETAILS_TEMPLATE; error_string = net::ErrorToString(error_code); DCHECK(error_code < 0); // Net error codes are negative. error_code = -error_code; } else if (error_domain == LocalizedError::kHttpErrorDomain) { error_page_template = IDS_ERRORPAGES_HTTP_DETAILS_TEMPLATE; error_string = HttpErrorToString(error_code); } else { NOTREACHED(); return string16(); } return l10n_util::GetStringFUTF16( error_page_template, base::IntToString16(error_code), ASCIIToUTF16(error_string), details); } const LocalizedErrorMap* FindErrorMapInArray(const LocalizedErrorMap* maps, size_t num_maps, int error_code) { for (size_t i = 0; i < num_maps; ++i) { if (maps[i].error_code == error_code) return &maps[i]; } return NULL; } const LocalizedErrorMap* LookupErrorMap(const std::string& error_domain, int error_code) { if (error_domain == net::kErrorDomain) { return FindErrorMapInArray(net_error_options, arraysize(net_error_options), error_code); } else if (error_domain == LocalizedError::kHttpErrorDomain) { return FindErrorMapInArray(http_error_options, arraysize(http_error_options), error_code); } else { NOTREACHED(); return NULL; } } bool LocaleIsRTL() { #if defined(TOOLKIT_GTK) // base::i18n::IsRTL() uses the GTK text direction, which doesn't work within // the renderer sandbox. return base::i18n::ICUIsRTL(); #else return base::i18n::IsRTL(); #endif } } // namespace const char LocalizedError::kHttpErrorDomain[] = "http"; void LocalizedError::GetStrings(const WebKit::WebURLError& error, DictionaryValue* error_strings) { bool rtl = LocaleIsRTL(); error_strings->SetString("textdirection", rtl ? "rtl" : "ltr"); // Grab strings that are applicable to all error pages error_strings->SetString("detailsLink", l10n_util::GetStringUTF16(IDS_ERRORPAGES_DETAILS_LINK)); error_strings->SetString("detailsHeading", l10n_util::GetStringUTF16(IDS_ERRORPAGES_DETAILS_HEADING)); // Grab the strings and settings that depend on the error type. Init // options with default values. LocalizedErrorMap options = { 0, IDS_ERRORPAGES_TITLE_NOT_AVAILABLE, IDS_ERRORPAGES_HEADING_NOT_AVAILABLE, IDS_ERRORPAGES_SUMMARY_NOT_AVAILABLE, IDS_ERRORPAGES_DETAILS_UNKNOWN, SUGGEST_NONE, }; const std::string error_domain = error.domain.utf8(); int error_code = error.reason; const LocalizedErrorMap* error_map = LookupErrorMap(error_domain, error_code); if (error_map) options = *error_map; string16 suggestions_heading; if (options.suggestions != SUGGEST_NONE) { suggestions_heading = l10n_util::GetStringUTF16(IDS_ERRORPAGES_SUGGESTION_HEADING); } error_strings->SetString("suggestionsHeading", suggestions_heading); string16 failed_url(ASCIIToUTF16(error.unreachableURL.spec())); // URLs are always LTR. if (rtl) base::i18n::WrapStringWithLTRFormatting(&failed_url); error_strings->SetString("title", l10n_util::GetStringFUTF16(options.title_resource_id, failed_url)); error_strings->SetString("heading", l10n_util::GetStringUTF16(options.heading_resource_id)); DictionaryValue* summary = new DictionaryValue; summary->SetString("msg", l10n_util::GetStringUTF16(options.summary_resource_id)); // TODO(tc): we want the unicode url here since it's being displayed summary->SetString("failedUrl", failed_url); error_strings->Set("summary", summary); string16 details = l10n_util::GetStringUTF16(options.details_resource_id); error_strings->SetString("details", GetErrorDetailsString(error_domain, error_code, details)); if (options.suggestions & SUGGEST_RELOAD) { DictionaryValue* suggest_reload = new DictionaryValue; suggest_reload->SetString("msg", l10n_util::GetStringUTF16(IDS_ERRORPAGES_SUGGESTION_RELOAD)); suggest_reload->SetString("reloadUrl", failed_url); error_strings->Set("suggestionsReload", suggest_reload); } if (options.suggestions & SUGGEST_HOSTNAME) { // Only show the "Go to hostname" suggestion if the failed_url has a path. const GURL& failed_url = error.unreachableURL; if (std::string() == failed_url.path()) { DictionaryValue* suggest_home_page = new DictionaryValue; suggest_home_page->SetString("suggestionsHomepageMsg", l10n_util::GetStringUTF16(IDS_ERRORPAGES_SUGGESTION_HOMEPAGE)); string16 homepage(ASCIIToUTF16(failed_url.GetWithEmptyPath().spec())); // URLs are always LTR. if (rtl) base::i18n::WrapStringWithLTRFormatting(&homepage); suggest_home_page->SetString("homePage", homepage); // TODO(tc): we actually want the unicode hostname suggest_home_page->SetString("hostName", failed_url.host()); error_strings->Set("suggestionsHomepage", suggest_home_page); } } if (options.suggestions & SUGGEST_LEARNMORE) { GURL learn_more_url; switch (options.error_code) { case net::ERR_TOO_MANY_REDIRECTS: learn_more_url = GURL(kRedirectLoopLearnMoreUrl); break; default: break; } if (learn_more_url.is_valid()) { // Add the language parameter to the URL. std::string query = learn_more_url.query() + "&hl=" + webkit_glue::GetWebKitLocale(); GURL::Replacements repl; repl.SetQueryStr(query); learn_more_url = learn_more_url.ReplaceComponents(repl); DictionaryValue* suggest_learn_more = new DictionaryValue; suggest_learn_more->SetString("msg", l10n_util::GetStringUTF16(IDS_ERRORPAGES_SUGGESTION_LEARNMORE)); suggest_learn_more->SetString("learnMoreUrl", learn_more_url.spec()); error_strings->Set("suggestionsLearnMore", suggest_learn_more); } } } bool LocalizedError::HasStrings(const std::string& error_domain, int error_code) { return LookupErrorMap(error_domain, error_code) != NULL; } void LocalizedError::GetFormRepostStrings(const GURL& display_url, DictionaryValue* error_strings) { bool rtl = LocaleIsRTL(); error_strings->SetString("textdirection", rtl ? "rtl" : "ltr"); string16 failed_url(ASCIIToUTF16(display_url.spec())); // URLs are always LTR. if (rtl) base::i18n::WrapStringWithLTRFormatting(&failed_url); error_strings->SetString( "title", l10n_util::GetStringFUTF16(IDS_ERRORPAGES_TITLE_NOT_AVAILABLE, failed_url)); error_strings->SetString( "heading", l10n_util::GetStringUTF16(IDS_HTTP_POST_WARNING_TITLE)); error_strings->SetString("suggestionsHeading", ""); DictionaryValue* summary = new DictionaryValue; summary->SetString( "msg", l10n_util::GetStringUTF16(IDS_ERRORPAGES_HTTP_POST_WARNING)); error_strings->Set("summary", summary); } void LocalizedError::GetAppErrorStrings(const WebURLError& error, const GURL& display_url, const ExtensionRendererInfo* app, DictionaryValue* error_strings) { DCHECK(app); bool rtl = LocaleIsRTL(); error_strings->SetString("textdirection", rtl ? "rtl" : "ltr"); string16 failed_url(ASCIIToUTF16(display_url.spec())); // URLs are always LTR. if (rtl) base::i18n::WrapStringWithLTRFormatting(&failed_url); error_strings->SetString( "url", l10n_util::GetStringFUTF16(IDS_ERRORPAGES_TITLE_NOT_AVAILABLE, failed_url.c_str())); error_strings->SetString("title", app->name()); error_strings->SetString("icon", app->icon_url().spec()); error_strings->SetString("name", app->name()); error_strings->SetString("msg", l10n_util::GetStringUTF16(IDS_ERRORPAGES_APP_WARNING)); }