From e81d4d7c8966f4650ea8041f8d0bad4fe72b4f54 Mon Sep 17 00:00:00 2001
From: "battre@chromium.org"
Date: Thu, 29 Sep 2011 16:54:31 +0000
Subject: Allow webRequest API extensions to clear the in-memory cache of
WebKit.
If the behavior of a webRequest API extension changes (i.e. different URLs are blocked) this effect was not visible instantly due to WebKit's in-memory cache. This CL introduces a means to clear the cache.
BUG=94284
TEST=no
Review URL: http://codereview.chromium.org/7835031
git-svn-id: svn://svn.chromium.org/chrome/trunk/src@103289 0039d316-1c4b-4281-b951-d872f2087c98
---
.../extensions/extension_function_dispatcher.cc | 1 +
.../browser/extensions/extension_webrequest_api.cc | 6 +
.../browser/extensions/extension_webrequest_api.h | 7 ++
chrome/browser/renderer_host/web_cache_manager.cc | 17 ++-
chrome/browser/renderer_host/web_cache_manager.h | 17 ++-
chrome/common/extensions/api/extension_api.json | 8 ++
.../extensions/docs/experimental.webRequest.html | 133 +++++++++++++++++++++
chrome/common/extensions/docs/samples.json | 1 +
chrome/common/render_messages.h | 3 +-
chrome/renderer/chrome_content_renderer_client.cc | 3 +-
chrome/renderer/chrome_render_process_observer.cc | 18 ++-
chrome/renderer/chrome_render_process_observer.h | 10 +-
chrome/renderer/chrome_render_view_observer.cc | 9 +-
chrome/renderer/chrome_render_view_observer.h | 7 +-
14 files changed, 226 insertions(+), 14 deletions(-)
diff --git a/chrome/browser/extensions/extension_function_dispatcher.cc b/chrome/browser/extensions/extension_function_dispatcher.cc
index 619d841..9d7b435 100644
--- a/chrome/browser/extensions/extension_function_dispatcher.cc
+++ b/chrome/browser/extensions/extension_function_dispatcher.cc
@@ -361,6 +361,7 @@ void FactoryRegistry::ResetFunctions() {
// WebRequest.
RegisterFunction();
RegisterFunction();
+ RegisterFunction();
// Preferences.
RegisterFunction();
diff --git a/chrome/browser/extensions/extension_webrequest_api.cc b/chrome/browser/extensions/extension_webrequest_api.cc
index cf13a52..1e80edd 100644
--- a/chrome/browser/extensions/extension_webrequest_api.cc
+++ b/chrome/browser/extensions/extension_webrequest_api.cc
@@ -20,6 +20,7 @@
#include "chrome/browser/extensions/extension_webrequest_time_tracker.h"
#include "chrome/browser/profiles/profile.h"
#include "chrome/browser/renderer_host/chrome_render_message_filter.h"
+#include "chrome/browser/renderer_host/web_cache_manager.h"
#include "chrome/common/extensions/extension.h"
#include "chrome/common/extensions/extension_error_utils.h"
#include "chrome/common/extensions/url_pattern.h"
@@ -1498,3 +1499,8 @@ bool WebRequestEventHandled::RunImpl() {
return true;
}
+
+bool WebRequestHandlerBehaviorChanged::RunImpl() {
+ WebCacheManager::GetInstance()->ClearCacheOnNavigation();
+ return true;
+}
diff --git a/chrome/browser/extensions/extension_webrequest_api.h b/chrome/browser/extensions/extension_webrequest_api.h
index f44d5b7..8a131c0 100644
--- a/chrome/browser/extensions/extension_webrequest_api.h
+++ b/chrome/browser/extensions/extension_webrequest_api.h
@@ -370,4 +370,11 @@ class WebRequestEventHandled : public SyncIOThreadExtensionFunction {
DECLARE_EXTENSION_FUNCTION_NAME("experimental.webRequest.eventHandled");
};
+class WebRequestHandlerBehaviorChanged : public AsyncExtensionFunction {
+ public:
+ virtual bool RunImpl();
+ DECLARE_EXTENSION_FUNCTION_NAME(
+ "experimental.webRequest.handlerBehaviorChanged");
+};
+
#endif // CHROME_BROWSER_EXTENSIONS_EXTENSION_WEBREQUEST_API_H_
diff --git a/chrome/browser/renderer_host/web_cache_manager.cc b/chrome/browser/renderer_host/web_cache_manager.cc
index 5a3a0ff..8ee268d 100644
--- a/chrome/browser/renderer_host/web_cache_manager.cc
+++ b/chrome/browser/renderer_host/web_cache_manager.cc
@@ -146,8 +146,15 @@ void WebCacheManager::SetGlobalSizeLimit(size_t bytes) {
void WebCacheManager::ClearCache() {
// Tell each renderer process to clear the cache.
- ClearRendederCache(active_renderers_);
- ClearRendederCache(inactive_renderers_);
+ ClearRendederCache(active_renderers_, INSTANTLY);
+ ClearRendederCache(inactive_renderers_, INSTANTLY);
+}
+
+void WebCacheManager::ClearCacheOnNavigation() {
+ // Tell each renderer process to clear the cache when a tab is reloaded or
+ // the user navigates to a new website.
+ ClearRendederCache(active_renderers_, ON_NAVIGATION);
+ ClearRendederCache(inactive_renderers_, ON_NAVIGATION);
}
void WebCacheManager::Observe(int type,
@@ -321,12 +328,14 @@ void WebCacheManager::EnactStrategy(const AllocationStrategy& strategy) {
}
}
-void WebCacheManager::ClearRendederCache(const std::set& renderers) {
+void WebCacheManager::ClearRendederCache(
+ const std::set& renderers,
+ WebCacheManager::ClearCacheOccasion occasion) {
std::set::const_iterator iter = renderers.begin();
for (; iter != renderers.end(); ++iter) {
RenderProcessHost* host = RenderProcessHost::FromID(*iter);
if (host)
- host->Send(new ChromeViewMsg_ClearCache());
+ host->Send(new ChromeViewMsg_ClearCache(occasion == ON_NAVIGATION));
}
}
diff --git a/chrome/browser/renderer_host/web_cache_manager.h b/chrome/browser/renderer_host/web_cache_manager.h
index 02c6958..0006572 100644
--- a/chrome/browser/renderer_host/web_cache_manager.h
+++ b/chrome/browser/renderer_host/web_cache_manager.h
@@ -1,4 +1,4 @@
-// Copyright (c) 2009 The Chromium Authors. All rights reserved.
+// Copyright (c) 2011 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.
@@ -70,6 +70,10 @@ class WebCacheManager : public NotificationObserver {
// Clears all in-memory caches.
void ClearCache();
+ // Clears all in-memory caches when a tab is reloaded or the user navigates
+ // to a different website.
+ void ClearCacheOnNavigation();
+
// NotificationObserver implementation:
virtual void Observe(int type,
const NotificationSource& source,
@@ -179,8 +183,17 @@ class WebCacheManager : public NotificationObserver {
// allocations according to |strategy|.
void EnactStrategy(const AllocationStrategy& strategy);
+ enum ClearCacheOccasion {
+ // Instructs to clear the cache instantly.
+ INSTANTLY,
+ // Instructs to clear the cache when a navigation takes place (this
+ // includes reloading a tab).
+ ON_NAVIGATION
+ };
+
// Inform all |renderers| to clear their cache.
- void ClearRendederCache(const std::set& renderers);
+ void ClearRendederCache(const std::set& renderers,
+ ClearCacheOccasion occation);
// Check to see if any active renderers have fallen inactive.
void FindInactiveRenderers();
diff --git a/chrome/common/extensions/api/extension_api.json b/chrome/common/extensions/api/extension_api.json
index 4f4077a..a0e91fe8 100644
--- a/chrome/common/extensions/api/extension_api.json
+++ b/chrome/common/extensions/api/extension_api.json
@@ -5704,6 +5704,14 @@
"name": "response"
}
]
+ },
+ {
+ "name": "handlerBehaviorChanged",
+ "type": "function",
+ "description": "Needs to be called when the behavior of the webRequest handlers has changed to prevent incorrect handling due to caching. This function call is expensive. Don't call it often.",
+ "parameters": [
+ {"type": "function", "name": "callback", "optional": true, "parameters": []}
+ ]
}
],
"events": [
diff --git a/chrome/common/extensions/docs/experimental.webRequest.html b/chrome/common/extensions/docs/experimental.webRequest.html
index 6b0847e..7b1f2cd 100644
--- a/chrome/common/extensions/docs/experimental.webRequest.html
+++ b/chrome/common/extensions/docs/experimental.webRequest.html
@@ -319,6 +319,8 @@
methodName
methodName
+
+ handlerBehaviorChanged
@@ -806,6 +808,137 @@ chrome.windows.onRemoved.addListener(
+
+
+
handlerBehaviorChanged
+
+
void
+
+ chrome.experimental.webRequest.handlerBehaviorChanged(, function
+ callback)
+
+
+
Undocumented.
+
Needs to be called when the behavior of the webRequest handlers has changed to prevent incorrect handling due to caching. This function call is expensive. Don't call it often.
+
+
+
Parameters
+
+
+
+
-
+ callback
+
+
+
+
+ (
+
optional
+
enumerated
+
+
+ Type
+
+
+
+ array of
+
+ function
+
+
+
+ )
+
+
+
+
+
-
+ Undocumented.
+
+
-
+ Description of this parameter from the json schema.
+
+
-
+ This parameter was added in version
+ .
+ You must omit this parameter in earlier versions,
+ and you may omit it in any version. If you require this
+ parameter, the manifest key
+ minimum_chrome_version
+ can ensure that your extension won't be run in an earlier browser version.
+
+
+
+
-
+
+
+
+
+
+
+
-
+
+
+
+
+
-
+
+
+
+
+
-
+
+
+
+
+
+
+
+
+
Returns
+
+
+
+
+
+
+
+
Callback function
+
+ The callback parameter should specify a function
+ that looks like this:
+
+
+ If you specify the callback parameter, it should
+ specify a function that looks like this:
+
+
+
+
function() {...};
+
+
+
+
+
+
+
+
+ This function was added in version .
+ If you require this function, the manifest key
+ minimum_chrome_version
+ can ensure that your extension won't be run in an earlier browser version.
+
+
+
diff --git a/chrome/common/extensions/docs/samples.json b/chrome/common/extensions/docs/samples.json
index 66f22ed..5614d6d 100644
--- a/chrome/common/extensions/docs/samples.json
+++ b/chrome/common/extensions/docs/samples.json
@@ -89,6 +89,7 @@
"chrome.experimental.webNavigation.onReferenceFragmentUpdated": "experimental.webNavigation.html#event-onReferenceFragmentUpdated",
"chrome.experimental.webRequest.addEventListener": "experimental.webRequest.html#method-addEventListener",
"chrome.experimental.webRequest.eventHandled": "experimental.webRequest.html#method-eventHandled",
+ "chrome.experimental.webRequest.handlerBehaviorChanged": "experimental.webRequest.html#method-handlerBehaviorChanged",
"chrome.experimental.webRequest.onAuthRequired": "experimental.webRequest.html#event-onAuthRequired",
"chrome.experimental.webRequest.onBeforeRedirect": "experimental.webRequest.html#event-onBeforeRedirect",
"chrome.experimental.webRequest.onBeforeRequest": "experimental.webRequest.html#event-onBeforeRequest",
diff --git a/chrome/common/render_messages.h b/chrome/common/render_messages.h
index 7d92fa2..84d1cff 100644
--- a/chrome/common/render_messages.h
+++ b/chrome/common/render_messages.h
@@ -138,7 +138,8 @@ IPC_MESSAGE_CONTROL3(ChromeViewMsg_SetCacheCapacities,
size_t /* capacity */)
// Tells the renderer to clear the cache.
-IPC_MESSAGE_CONTROL0(ChromeViewMsg_ClearCache)
+IPC_MESSAGE_CONTROL1(ChromeViewMsg_ClearCache,
+ bool /* on_navigation */)
// Tells the renderer to dump as much memory as it can, perhaps because we
// have memory pressure or the renderer is (or will be) paged out. This
diff --git a/chrome/renderer/chrome_content_renderer_client.cc b/chrome/renderer/chrome_content_renderer_client.cc
index 91841cf..88d41cf 100644
--- a/chrome/renderer/chrome_content_renderer_client.cc
+++ b/chrome/renderer/chrome_content_renderer_client.cc
@@ -243,7 +243,8 @@ void ChromeContentRendererClient::RenderViewCreated(RenderView* render_view) {
TranslateHelper* translate = new TranslateHelper(render_view, autofill_agent);
new ChromeRenderViewObserver(
- render_view, content_settings, extension_dispatcher_.get(), translate);
+ render_view, content_settings, chrome_observer_.get(),
+ extension_dispatcher_.get(), translate);
// Used only for testing/automation.
if (CommandLine::ForCurrentProcess()->HasSwitch(
diff --git a/chrome/renderer/chrome_render_process_observer.cc b/chrome/renderer/chrome_render_process_observer.cc
index 93d1fa8..ad41476 100644
--- a/chrome/renderer/chrome_render_process_observer.cc
+++ b/chrome/renderer/chrome_render_process_observer.cc
@@ -196,7 +196,8 @@ bool ChromeRenderProcessObserver::is_incognito_process_ = false;
ChromeRenderProcessObserver::ChromeRenderProcessObserver(
chrome::ChromeContentRendererClient* client)
- : client_(client) {
+ : client_(client),
+ clear_cache_pending_(false) {
const CommandLine& command_line = *CommandLine::ForCurrentProcess();
if (command_line.HasSwitch(switches::kEnableWatchdog)) {
// TODO(JAR): Need to implement renderer IO msgloop watchdog.
@@ -314,8 +315,12 @@ void ChromeRenderProcessObserver::OnSetCacheCapacities(size_t min_dead_capacity,
min_dead_capacity, max_dead_capacity, capacity);
}
-void ChromeRenderProcessObserver::OnClearCache() {
- WebCache::clear();
+void ChromeRenderProcessObserver::OnClearCache(bool on_navigation) {
+ if (on_navigation) {
+ clear_cache_pending_ = true;
+ } else {
+ WebCache::clear();
+ }
}
void ChromeRenderProcessObserver::OnGetCacheResourceStats() {
@@ -409,3 +414,10 @@ void ChromeRenderProcessObserver::OnPurgeMemory() {
if (client_)
client_->OnPurgeMemory();
}
+
+void ChromeRenderProcessObserver::ExecutePendingClearCache() {
+ if (clear_cache_pending_) {
+ clear_cache_pending_ = false;
+ WebCache::clear();
+ }
+}
diff --git a/chrome/renderer/chrome_render_process_observer.h b/chrome/renderer/chrome_render_process_observer.h
index 80bf18ef..615b6c3 100644
--- a/chrome/renderer/chrome_render_process_observer.h
+++ b/chrome/renderer/chrome_render_process_observer.h
@@ -33,6 +33,10 @@ class ChromeRenderProcessObserver : public RenderProcessObserver {
static bool is_incognito_process() { return is_incognito_process_; }
+ // Needs to be called by RenderViews in case of navigations to execute
+ // any 'clear cache' commands that were delayed until the next navigation.
+ void ExecutePendingClearCache();
+
private:
// RenderProcessObserver implementation.
virtual bool OnControlMessageReceived(const IPC::Message& message) OVERRIDE;
@@ -45,7 +49,9 @@ class ChromeRenderProcessObserver : public RenderProcessObserver {
void OnSetCacheCapacities(size_t min_dead_capacity,
size_t max_dead_capacity,
size_t capacity);
- void OnClearCache();
+ // If |on_navigation| is true, the clearing is delayed until the next
+ // navigation event.
+ void OnClearCache(bool on_navigation);
void OnGetCacheResourceStats();
void OnSetFieldTrialGroup(const std::string& fiel_trial_name,
const std::string& group_name);
@@ -58,6 +64,8 @@ class ChromeRenderProcessObserver : public RenderProcessObserver {
static bool is_incognito_process_;
scoped_ptr resource_delegate_;
chrome::ChromeContentRendererClient* client_;
+ // If true, the web cache shall be cleared before the next navigation event.
+ bool clear_cache_pending_;
DISALLOW_COPY_AND_ASSIGN(ChromeRenderProcessObserver);
};
diff --git a/chrome/renderer/chrome_render_view_observer.cc b/chrome/renderer/chrome_render_view_observer.cc
index b913137..f4648fbc 100644
--- a/chrome/renderer/chrome_render_view_observer.cc
+++ b/chrome/renderer/chrome_render_view_observer.cc
@@ -16,6 +16,7 @@
#include "chrome/common/url_constants.h"
#include "chrome/renderer/about_handler.h"
#include "chrome/renderer/automation/dom_automation_controller.h"
+#include "chrome/renderer/chrome_render_process_observer.h"
#include "chrome/renderer/content_settings_observer.h"
#include "chrome/renderer/extensions/extension_dispatcher.h"
#include "chrome/renderer/external_host_bindings.h"
@@ -198,11 +199,13 @@ GURL StripRef(const GURL& url) {
ChromeRenderViewObserver::ChromeRenderViewObserver(
RenderView* render_view,
ContentSettingsObserver* content_settings,
+ ChromeRenderProcessObserver* chrome_render_process_observer,
ExtensionDispatcher* extension_dispatcher,
TranslateHelper* translate_helper)
: RenderViewObserver(render_view),
- content_settings_(content_settings),
+ chrome_render_process_observer_(chrome_render_process_observer),
extension_dispatcher_(extension_dispatcher),
+ content_settings_(content_settings),
translate_helper_(translate_helper),
phishing_classifier_(NULL),
last_indexed_page_id_(-1),
@@ -336,6 +339,10 @@ void ChromeRenderViewObserver::OnSetAllowRunningInsecureContent(bool allow) {
}
void ChromeRenderViewObserver::Navigate(const GURL& url) {
+ // Execute cache clear operations that were postponed until a navigation
+ // event (including tab reload).
+ if (chrome_render_process_observer_)
+ chrome_render_process_observer_->ExecutePendingClearCache();
AboutHandler::MaybeHandle(url);
}
diff --git a/chrome/renderer/chrome_render_view_observer.h b/chrome/renderer/chrome_render_view_observer.h
index fa2946a..cf1497e 100644
--- a/chrome/renderer/chrome_render_view_observer.h
+++ b/chrome/renderer/chrome_render_view_observer.h
@@ -16,6 +16,7 @@
#include "googleurl/src/gurl.h"
#include "third_party/WebKit/Source/WebKit/chromium/public/WebPermissionClient.h"
+class ChromeRenderProcessObserver;
class ContentSettingsObserver;
class DomAutomationController;
class ExtensionDispatcher;
@@ -43,6 +44,7 @@ class ChromeRenderViewObserver : public RenderViewObserver,
ChromeRenderViewObserver(
RenderView* render_view,
ContentSettingsObserver* content_settings,
+ ChromeRenderProcessObserver* chrome_render_process_observer,
ExtensionDispatcher* extension_dispatcher,
TranslateHelper* translate_helper);
virtual ~ChromeRenderViewObserver();
@@ -171,9 +173,12 @@ class ChromeRenderViewObserver : public RenderViewObserver,
// Save the JavaScript to preload if a ViewMsg_WebUIJavaScript is received.
scoped_ptr webui_javascript_;
+ // Owned by ChromeContentRendererClient and outlive us.
+ ChromeRenderProcessObserver* chrome_render_process_observer_;
+ ExtensionDispatcher* extension_dispatcher_;
+
// Have the same lifetime as us.
ContentSettingsObserver* content_settings_;
- ExtensionDispatcher* extension_dispatcher_;
TranslateHelper* translate_helper_;
safe_browsing::PhishingClassifierDelegate* phishing_classifier_;
--
cgit v1.1