/*
* 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