Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(1058)

Unified Diff: chrome/browser/extensions/extension_warning_set.cc

Issue 10407105: Improve error messaging of webRequest API in case of conflicts (Closed) Base URL: svn://svn.chromium.org/chrome/trunk/src
Patch Set: Addressed Evan's comments Created 8 years, 1 month ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
Index: chrome/browser/extensions/extension_warning_set.cc
diff --git a/chrome/browser/extensions/extension_warning_set.cc b/chrome/browser/extensions/extension_warning_set.cc
index 2d73cd684c4b3ea58c9a276ba879757399b98583..fb18f91af01f1e8b48f910e7147318420574d408 100644
--- a/chrome/browser/extensions/extension_warning_set.cc
+++ b/chrome/browser/extensions/extension_warning_set.cc
@@ -4,192 +4,204 @@
#include "chrome/browser/extensions/extension_warning_set.h"
+#include "base/utf_string_conversions.h"
#include "chrome/browser/browser_process.h"
-#include "chrome/browser/extensions/extension_global_error_badge.h"
-#include "chrome/browser/extensions/extension_service.h"
-#include "chrome/browser/profiles/profile.h"
-#include "chrome/browser/profiles/profile_manager.h"
-#include "chrome/browser/ui/global_error/global_error_service.h"
-#include "chrome/browser/ui/global_error/global_error_service_factory.h"
#include "chrome/common/chrome_notification_types.h"
+#include "chrome/common/extensions/extension.h"
+#include "chrome/common/extensions/extension_set.h"
#include "content/public/browser/browser_thread.h"
-#include "content/public/browser/notification_service.h"
#include "grit/chromium_strings.h"
#include "grit/generated_resources.h"
+#include "net/base/escape.h"
#include "ui/base/l10n/l10n_util.h"
using content::BrowserThread;
-// This class is used to represent warnings if extensions misbehave.
-class ExtensionWarning {
- public:
- // Default constructor for storing ExtensionServiceWarning in STL containers
- // do not use.
- ExtensionWarning();
-
- // Constructs a warning of type |type| for extension |extension_id|. This
- // could be for example the fact that an extension conflicted with others.
- ExtensionWarning(ExtensionWarningSet::WarningType type,
- const std::string& extension_id);
-
- ~ExtensionWarning();
-
- // Returns the specific warning type.
- ExtensionWarningSet::WarningType warning_type() const { return type_; }
-
- // Returns the id of the extension for which this warning is valid.
- const std::string& extension_id() const { return extension_id_; }
+namespace {
+// Prefix for message parameters indicating that the parameter needs to
+// be translated from an extension id to the extension name.
+const char kTranslate[] = "TO_TRANSLATE:";
+const size_t kMaxNumberOfParameters = 4;
+}
- private:
- ExtensionWarningSet::WarningType type_;
- std::string extension_id_;
+namespace extensions {
- // Allow implicit copy and assign operator.
-};
-
-ExtensionWarning::ExtensionWarning() : type_(ExtensionWarningSet::kInvalid) {
-}
+//
+// ExtensionWarning
+//
ExtensionWarning::ExtensionWarning(
- ExtensionWarningSet::WarningType type,
- const std::string& extension_id)
- : type_(type), extension_id_(extension_id) {
+ WarningType type,
+ const std::string& extension_id,
+ int message_id,
+ const std::vector<std::string>& message_parameters)
+ : type_(type),
+ extension_id_(extension_id),
+ message_id_(message_id),
+ message_parameters_(message_parameters) {
// These are invalid here because they do not have corresponding warning
// messages in the UI.
- CHECK(type != ExtensionWarningSet::kInvalid);
- CHECK(type != ExtensionWarningSet::kMaxWarningType);
-}
-
-ExtensionWarning::~ExtensionWarning() {
+ CHECK_NE(type, kInvalid);
+ CHECK_NE(type, kMaxWarningType);
+ CHECK_LE(message_parameters.size(), kMaxNumberOfParameters);
}
-bool operator<(const ExtensionWarning& a, const ExtensionWarning& b) {
- if (a.warning_type() == b.warning_type())
- return a.extension_id() < b.extension_id();
- return a.warning_type() < b.warning_type();
-}
+ExtensionWarning::ExtensionWarning(const ExtensionWarning& other)
+ : type_(other.type_),
+ extension_id_(other.extension_id_),
+ message_id_(other.message_id_),
+ message_parameters_(other.message_parameters_) {}
-// Static
-string16 ExtensionWarningSet::GetLocalizedWarning(
- ExtensionWarningSet::WarningType warning_type) {
- switch (warning_type) {
- case kInvalid:
- case kMaxWarningType:
- NOTREACHED();
- return string16();
- case kNetworkDelay:
- return l10n_util::GetStringFUTF16(
- IDS_EXTENSION_WARNINGS_NETWORK_DELAY,
- l10n_util::GetStringUTF16(IDS_PRODUCT_NAME));
- case kNetworkConflict:
- return l10n_util::GetStringUTF16(IDS_EXTENSION_WARNINGS_NETWORK_CONFLICT);
- case kRepeatedCacheFlushes:
- return l10n_util::GetStringFUTF16(
- IDS_EXTENSION_WARNINGS_NETWORK_DELAY,
- l10n_util::GetStringUTF16(IDS_PRODUCT_NAME));
- }
- NOTREACHED(); // Switch statement has no default branch.
- return string16();
+ExtensionWarning::~ExtensionWarning() {
}
-ExtensionWarningSet::ExtensionWarningSet(Profile* profile) : profile_(profile) {
+ExtensionWarning& ExtensionWarning::operator=(const ExtensionWarning& other) {
+ type_ = other.type_;
+ extension_id_ = other.extension_id_;
+ message_id_ = other.message_id_;
+ message_parameters_ = other.message_parameters_;
+ return *this;
}
-ExtensionWarningSet::~ExtensionWarningSet() {
+// static
+ExtensionWarning ExtensionWarning::CreateNetworkDelayWarning(
+ const std::string& extension_id) {
+ std::vector<std::string> message_parameters;
+ message_parameters.push_back(l10n_util::GetStringUTF8(IDS_PRODUCT_NAME));
+ return ExtensionWarning(
+ kNetworkDelay,
+ extension_id,
+ IDS_EXTENSION_WARNINGS_NETWORK_DELAY,
+ message_parameters);
}
-void ExtensionWarningSet::SetWarning(ExtensionWarningSet::WarningType type,
- const std::string& extension_id) {
- ExtensionWarning warning(type, extension_id);
- bool inserted = warnings_.insert(warning).second;
- if (inserted) {
- NotifyWarningsChanged();
- UpdateWarningBadge();
- }
+// static
+ExtensionWarning ExtensionWarning::CreateNetworkConflictWarning(
+ const std::string& extension_id) {
+ std::vector<std::string> message_parameters;
+ return ExtensionWarning(
+ kNetworkConflict,
+ extension_id,
+ IDS_EXTENSION_WARNINGS_NETWORK_CONFLICT,
+ message_parameters);
}
-void ExtensionWarningSet::ClearWarnings(
- const std::set<ExtensionWarningSet::WarningType>& types) {
- bool deleted_anything = false;
- for (iterator i = warnings_.begin(); i != warnings_.end();) {
- if (types.find(i->warning_type()) != types.end()) {
- deleted_anything = true;
- warnings_.erase(i++);
- } else {
- ++i;
- }
- }
-
- if (deleted_anything) {
- NotifyWarningsChanged();
- UpdateWarningBadge();
- }
+// static
+ExtensionWarning ExtensionWarning::CreateRedirectConflictWarning(
+ const std::string& extension_id,
+ const std::string& winning_extension_id,
+ const GURL& attempted_redirect_url,
+ const GURL& winning_redirect_url) {
+ std::vector<std::string> message_parameters;
+ message_parameters.push_back(attempted_redirect_url.spec());
+ message_parameters.push_back(kTranslate + winning_extension_id);
+ message_parameters.push_back(winning_redirect_url.spec());
+ return ExtensionWarning(
+ kRedirectConflict,
+ extension_id,
+ IDS_EXTENSION_WARNINGS_REDIRECT_CONFLICT,
+ message_parameters);
}
-void ExtensionWarningSet::GetWarningsAffectingExtension(
+// static
+ExtensionWarning ExtensionWarning::CreateRequestHeaderConflictWarning(
const std::string& extension_id,
- std::set<ExtensionWarningSet::WarningType>* result) const {
- result->clear();
- for (const_iterator i = warnings_.begin(); i != warnings_.end(); ++i) {
- if (i->extension_id() == extension_id)
- result->insert(i->warning_type());
- }
+ const std::string& winning_extension_id,
+ const std::string& conflicting_header) {
+ std::vector<std::string> message_parameters;
+ message_parameters.push_back(conflicting_header);
+ message_parameters.push_back(kTranslate + winning_extension_id);
+ return ExtensionWarning(
+ kNetworkConflict,
+ extension_id,
+ IDS_EXTENSION_WARNINGS_REQUEST_HEADER_CONFLICT,
+ message_parameters);
}
// static
-void ExtensionWarningSet::NotifyWarningsOnUI(
- void* profile_id,
- std::set<std::string> extension_ids,
- WarningType warning_type) {
- CHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
- Profile* profile = reinterpret_cast<Profile*>(profile_id);
- if (!profile ||
- !g_browser_process->profile_manager() ||
- !g_browser_process->profile_manager()->IsValidProfile(profile)) {
- return;
- }
-
- ExtensionWarningSet* warnings =
- profile->GetExtensionService()->extension_warnings();
-
- for (std::set<std::string>::const_iterator i = extension_ids.begin();
- i != extension_ids.end(); ++i) {
- warnings->SetWarning(warning_type, *i);
- }
+ExtensionWarning ExtensionWarning::CreateResponseHeaderConflictWarning(
+ const std::string& extension_id,
+ const std::string& winning_extension_id,
+ const std::string& conflicting_header) {
+ std::vector<std::string> message_parameters;
+ message_parameters.push_back(conflicting_header);
+ message_parameters.push_back(kTranslate + winning_extension_id);
+ return ExtensionWarning(
+ kNetworkConflict,
+ extension_id,
+ IDS_EXTENSION_WARNINGS_RESPONSE_HEADER_CONFLICT,
+ message_parameters);
}
-void ExtensionWarningSet::SuppressBadgeForCurrentWarnings() {
- badge_suppressions_.insert(warnings_.begin(), warnings_.end());
- UpdateWarningBadge();
+// static
+ExtensionWarning ExtensionWarning::CreateCredentialsConflictWarning(
+ const std::string& extension_id,
+ const std::string& winning_extension_id) {
+ std::vector<std::string> message_parameters;
+ message_parameters.push_back(kTranslate + winning_extension_id);
+ return ExtensionWarning(
+ kNetworkConflict,
+ extension_id,
+ IDS_EXTENSION_WARNINGS_CREDENTIALS_CONFLICT,
+ message_parameters);
}
-void ExtensionWarningSet::NotifyWarningsChanged() {
- content::NotificationService::current()->Notify(
- chrome::NOTIFICATION_EXTENSION_WARNING_CHANGED,
- content::Source<Profile>(profile_),
- content::NotificationService::NoDetails());
+// static
+ExtensionWarning ExtensionWarning::CreateRepeatedCacheFlushesWarning(
+ const std::string& extension_id) {
+ std::vector<std::string> message_parameters;
+ message_parameters.push_back(l10n_util::GetStringUTF8(IDS_PRODUCT_NAME));
+ return ExtensionWarning(
+ kRepeatedCacheFlushes,
+ extension_id,
+ IDS_EXTENSION_WARNINGS_NETWORK_DELAY,
+ message_parameters);
}
-void ExtensionWarningSet::UpdateWarningBadge() {
- // We need a badge if a warning exists that has not been suppressed.
- bool need_warning_badge = false;
- for (const_iterator i = warnings_.begin(); i != warnings_.end(); ++i) {
- if (badge_suppressions_.find(*i) == badge_suppressions_.end()) {
- need_warning_badge = true;
- break;
+std::string ExtensionWarning::GetLocalizedMessage(
+ const ExtensionSet* extensions) const {
+ DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
+
+ // These parameters may be unsafe (URLs and Extension names) and need
+ // to be HTML-escaped before being embedded in the UI. Also extension IDs
+ // are translated to full extension names.
+ std::vector<string16> final_parameters;
+ for (size_t i = 0; i < message_parameters_.size(); ++i) {
+ std::string message = message_parameters_[i];
+ if (StartsWithASCII(message, kTranslate, true)) {
+ std::string extension_id = message.substr(sizeof(kTranslate) - 1);
+ const extensions::Extension* extension =
+ extensions->GetByID(extension_id);
+ message = extension ? extension->name() : extension_id;
}
+ final_parameters.push_back(UTF8ToUTF16(net::EscapeForHTML(message)));
}
- GlobalErrorService* service =
- GlobalErrorServiceFactory::GetForProfile(profile_);
- GlobalError* error = service->GetGlobalErrorByMenuItemCommandID(
- ExtensionGlobalErrorBadge::GetMenuItemCommandID());
-
- // Activate or hide the warning badge in case the current state is incorrect.
- if (error && !need_warning_badge) {
- service->RemoveGlobalError(error);
- delete error;
- } else if (!error && need_warning_badge) {
- service->AddGlobalError(new ExtensionGlobalErrorBadge);
+ COMPILE_ASSERT(kMaxNumberOfParameters == 4u, YouNeedToAddMoreCaseStatements);
+ switch (final_parameters.size()) {
+ case 0:
+ return l10n_util::GetStringUTF8(message_id_);
+ case 1:
+ return l10n_util::GetStringFUTF8(message_id_, final_parameters[0]);
+ case 2:
+ return l10n_util::GetStringFUTF8(message_id_, final_parameters[0],
+ final_parameters[1]);
+ case 3:
+ return l10n_util::GetStringFUTF8(message_id_, final_parameters[0],
+ final_parameters[1], final_parameters[2]);
+ case 4:
+ return l10n_util::GetStringFUTF8(message_id_, final_parameters[0],
+ final_parameters[1], final_parameters[2], final_parameters[3]);
+ default:
+ NOTREACHED();
+ return std::string();
}
}
+
+bool operator<(const ExtensionWarning& a, const ExtensionWarning& b) {
+ if (a.extension_id() != b.extension_id())
+ return a.extension_id() < b.extension_id();
+ return a.warning_type() < b.warning_type();
+}
+
+} // namespace extensions

Powered by Google App Engine
This is Rietveld 408576698