| /* |
| * Copyright (C) 2011 Apple Inc. All rights reserved. |
| * |
| * Redistribution and use in source and binary forms, with or without |
| * modification, are permitted provided that the following conditions |
| * are met: |
| * 1. Redistributions of source code must retain the above copyright |
| * notice, this list of conditions and the following disclaimer. |
| * 2. Redistributions in binary form must reproduce the above copyright |
| * notice, this list of conditions and the following disclaimer in the |
| * documentation and/or other materials provided with the distribution. |
| * |
| * THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS'' |
| * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, |
| * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR |
| * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS |
| * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR |
| * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF |
| * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS |
| * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN |
| * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) |
| * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF |
| * THE POSSIBILITY OF SUCH DAMAGE. |
| */ |
| |
| #include "config.h" |
| #include "WebPluginSiteDataManager.h" |
| |
| #include "ImmutableArray.h" |
| #include "PluginProcessManager.h" |
| #include "WebContext.h" |
| #include "WebProcessMessages.h" |
| |
| using namespace WebCore; |
| |
| namespace WebKit { |
| |
| #if ENABLE(PLUGIN_PROCESS) |
| class WebPluginSiteDataManager::GetSitesWithDataState { |
| public: |
| explicit GetSitesWithDataState(WebPluginSiteDataManager* webPluginSiteDataManager, uint64_t callbackID) |
| : m_webPluginSiteDataManager(webPluginSiteDataManager) |
| , m_callbackID(callbackID) |
| , m_plugins(webPluginSiteDataManager->m_webContext->pluginInfoStore()->plugins()) |
| { |
| } |
| |
| void getSitesWithDataForNextPlugin() |
| { |
| if (m_plugins.isEmpty()) { |
| Vector<String> sites; |
| copyToVector(m_sites, sites); |
| |
| m_webPluginSiteDataManager->didGetSitesWithDataForAllPlugins(sites, m_callbackID); |
| return; |
| } |
| |
| PluginProcessManager::shared().getSitesWithData(m_plugins.last(), m_webPluginSiteDataManager, m_callbackID); |
| m_plugins.removeLast(); |
| } |
| |
| void didGetSitesWithDataForSinglePlugin(const Vector<String>& sites) |
| { |
| for (size_t i = 0; i < sites.size(); ++i) |
| m_sites.add(sites[i]); |
| |
| getSitesWithDataForNextPlugin(); |
| } |
| |
| private: |
| WebPluginSiteDataManager* m_webPluginSiteDataManager; |
| uint64_t m_callbackID; |
| Vector<PluginInfoStore::Plugin> m_plugins; |
| HashSet<String> m_sites; |
| }; |
| |
| class WebPluginSiteDataManager::ClearSiteDataState { |
| public: |
| explicit ClearSiteDataState(WebPluginSiteDataManager* webPluginSiteDataManager, const Vector<String>& sites, uint64_t flags, uint64_t maxAgeInSeconds, uint64_t callbackID) |
| : m_webPluginSiteDataManager(webPluginSiteDataManager) |
| , m_sites(sites) |
| , m_flags(flags) |
| , m_maxAgeInSeconds(maxAgeInSeconds) |
| , m_callbackID(callbackID) |
| , m_plugins(webPluginSiteDataManager->m_webContext->pluginInfoStore()->plugins()) |
| { |
| } |
| |
| void clearSiteDataForNextPlugin() |
| { |
| if (m_plugins.isEmpty()) { |
| m_webPluginSiteDataManager->didClearSiteDataForAllPlugins(m_callbackID); |
| return; |
| } |
| |
| PluginProcessManager::shared().clearSiteData(m_plugins.last(), m_webPluginSiteDataManager, m_sites, m_flags, m_maxAgeInSeconds, m_callbackID); |
| m_plugins.removeLast(); |
| } |
| |
| void didClearSiteDataForSinglePlugin() |
| { |
| clearSiteDataForNextPlugin(); |
| } |
| |
| private: |
| WebPluginSiteDataManager* m_webPluginSiteDataManager; |
| Vector<String> m_sites; |
| uint64_t m_flags; |
| uint64_t m_maxAgeInSeconds; |
| uint64_t m_callbackID; |
| Vector<PluginInfoStore::Plugin> m_plugins; |
| }; |
| #endif // ENABLE(PLUGIN_PROCESS) |
| |
| PassRefPtr<WebPluginSiteDataManager> WebPluginSiteDataManager::create(WebContext* webContext) |
| { |
| return adoptRef(new WebPluginSiteDataManager(webContext)); |
| } |
| |
| WebPluginSiteDataManager::WebPluginSiteDataManager(WebContext* webContext) |
| : m_webContext(webContext) |
| { |
| } |
| |
| WebPluginSiteDataManager::~WebPluginSiteDataManager() |
| { |
| ASSERT(m_arrayCallbacks.isEmpty()); |
| ASSERT(m_voidCallbacks.isEmpty()); |
| #if ENABLE(PLUGIN_PROCESS) |
| ASSERT(m_pendingGetSitesWithData.isEmpty()); |
| ASSERT(m_pendingClearSiteData.isEmpty()); |
| #endif |
| } |
| |
| void WebPluginSiteDataManager::invalidate() |
| { |
| invalidateCallbackMap(m_arrayCallbacks); |
| |
| #if ENABLE(PLUGIN_PROCESS) |
| deleteAllValues(m_pendingGetSitesWithData); |
| m_pendingGetSitesWithData.clear(); |
| deleteAllValues(m_pendingClearSiteData); |
| m_pendingClearSiteData.clear(); |
| #endif |
| } |
| |
| void WebPluginSiteDataManager::getSitesWithData(PassRefPtr<ArrayCallback> prpCallback) |
| { |
| RefPtr<ArrayCallback> callback = prpCallback; |
| |
| if (!m_webContext) { |
| callback->invalidate(); |
| return; |
| } |
| |
| uint64_t callbackID = callback->callbackID(); |
| m_arrayCallbacks.set(callbackID, callback.release()); |
| |
| #if ENABLE(PLUGIN_PROCESS) |
| ASSERT(!m_pendingGetSitesWithData.contains(callbackID)); |
| |
| GetSitesWithDataState* state = new GetSitesWithDataState(this, callbackID); |
| m_pendingGetSitesWithData.set(callbackID, state); |
| state->getSitesWithDataForNextPlugin(); |
| #else |
| m_webContext->relaunchProcessIfNecessary(); |
| |
| Vector<String> pluginPaths; |
| m_webContext->pluginInfoStore()->getPluginPaths(pluginPaths); |
| |
| // FIXME (Multi-WebProcess): When multi-process is enabled, we must always use a plug-in process for this, |
| // so this code should just be removed. |
| m_webContext->sendToAllProcessesRelaunchingThemIfNecessary(Messages::WebProcess::GetSitesWithPluginData(pluginPaths, callbackID)); |
| #endif |
| } |
| |
| void WebPluginSiteDataManager::didGetSitesWithData(const Vector<String>& sites, uint64_t callbackID) |
| { |
| RefPtr<ArrayCallback> callback = m_arrayCallbacks.take(callbackID); |
| if (!callback) { |
| // FIXME: Log error or assert. |
| return; |
| } |
| |
| Vector<RefPtr<APIObject> > sitesWK(sites.size()); |
| |
| for (size_t i = 0; i < sites.size(); ++i) |
| sitesWK[i] = WebString::create(sites[i]); |
| |
| RefPtr<ImmutableArray> resultArray = ImmutableArray::adopt(sitesWK); |
| callback->performCallbackWithReturnValue(resultArray.get()); |
| } |
| |
| void WebPluginSiteDataManager::clearSiteData(ImmutableArray* sites, uint64_t flags, uint64_t maxAgeInSeconds, PassRefPtr<VoidCallback> prpCallback) |
| { |
| RefPtr<VoidCallback> callback = prpCallback; |
| if (!m_webContext) { |
| callback->invalidate(); |
| return; |
| } |
| |
| Vector<String> sitesVector; |
| |
| // If the array is empty, don't do anything. |
| if (sites) { |
| if (!sites->size()) { |
| callback->performCallback(); |
| return; |
| } |
| |
| for (size_t i = 0; i < sites->size(); ++i) { |
| if (WebString* site = sites->at<WebString>(i)) |
| sitesVector.append(site->string()); |
| } |
| } |
| |
| uint64_t callbackID = callback->callbackID(); |
| m_voidCallbacks.set(callbackID, callback.release()); |
| |
| #if ENABLE(PLUGIN_PROCESS) |
| ASSERT(!m_pendingClearSiteData.contains(callbackID)); |
| |
| ClearSiteDataState* state = new ClearSiteDataState(this, sitesVector, flags, maxAgeInSeconds, callbackID); |
| m_pendingClearSiteData.set(callbackID, state); |
| state->clearSiteDataForNextPlugin(); |
| #else |
| |
| m_webContext->relaunchProcessIfNecessary(); |
| Vector<String> pluginPaths; |
| m_webContext->pluginInfoStore()->getPluginPaths(pluginPaths); |
| |
| // FIXME (Multi-WebProcess): When multi-process is enabled, we must always use a plug-in process for this, |
| // so this code should just be removed. |
| m_webContext->sendToAllProcessesRelaunchingThemIfNecessary(Messages::WebProcess::ClearPluginSiteData(pluginPaths, sitesVector, flags, maxAgeInSeconds, callbackID)); |
| #endif |
| } |
| |
| void WebPluginSiteDataManager::didClearSiteData(uint64_t callbackID) |
| { |
| RefPtr<VoidCallback> callback = m_voidCallbacks.take(callbackID); |
| if (!callback) { |
| // FIXME: Log error or assert. |
| return; |
| } |
| |
| callback->performCallback(); |
| } |
| |
| bool WebPluginSiteDataManager::shouldTerminate(WebProcessProxy*) const |
| { |
| #if ENABLE(PLUGIN_PROCESS) |
| // When out of process plug-ins are enabled, the web process is not involved in fetching site data. |
| return true; |
| #else |
| return m_arrayCallbacks.isEmpty() && m_voidCallbacks.isEmpty(); |
| #endif |
| } |
| |
| #if ENABLE(PLUGIN_PROCESS) |
| void WebPluginSiteDataManager::didGetSitesWithDataForSinglePlugin(const Vector<String>& sites, uint64_t callbackID) |
| { |
| GetSitesWithDataState* state = m_pendingGetSitesWithData.get(callbackID); |
| ASSERT(state); |
| |
| state->didGetSitesWithDataForSinglePlugin(sites); |
| } |
| |
| void WebPluginSiteDataManager::didGetSitesWithDataForAllPlugins(const Vector<String>& sites, uint64_t callbackID) |
| { |
| OwnPtr<GetSitesWithDataState> state = adoptPtr(m_pendingGetSitesWithData.take(callbackID)); |
| ASSERT(state); |
| |
| didGetSitesWithData(sites, callbackID); |
| } |
| |
| void WebPluginSiteDataManager::didClearSiteDataForSinglePlugin(uint64_t callbackID) |
| { |
| ClearSiteDataState* state = m_pendingClearSiteData.get(callbackID); |
| ASSERT(state); |
| |
| state->didClearSiteDataForSinglePlugin(); |
| } |
| |
| void WebPluginSiteDataManager::didClearSiteDataForAllPlugins(uint64_t callbackID) |
| { |
| OwnPtr<ClearSiteDataState> state = adoptPtr(m_pendingClearSiteData.take(callbackID)); |
| ASSERT(state); |
| |
| didClearSiteData(callbackID); |
| } |
| |
| #endif |
| |
| } // namespace WebKit |
| |