blob: b022a896e6abd30189e2eea542fb8e897e4cf220 [file] [log] [blame]
/*
* Copyright (C) 2013 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 "StorageAreaImpl.h"
#include "StorageAreaMap.h"
#include <WebCore/Document.h>
#include <WebCore/ExceptionCode.h>
#include <WebCore/Frame.h>
#include <WebCore/Page.h>
#include <WebCore/SchemeRegistry.h>
#include <WebCore/Settings.h>
using namespace WebCore;
namespace WebKit {
static uint64_t generateStorageAreaID()
{
static uint64_t storageAreaID;
return ++storageAreaID;
}
PassRefPtr<StorageAreaImpl> StorageAreaImpl::create(PassRefPtr<StorageAreaMap> storageAreaMap)
{
return adoptRef(new StorageAreaImpl(storageAreaMap));
}
StorageAreaImpl::StorageAreaImpl(PassRefPtr<StorageAreaMap> storageAreaMap)
: m_storageAreaID(generateStorageAreaID())
, m_storageAreaMap(storageAreaMap)
{
}
StorageAreaImpl::~StorageAreaImpl()
{
}
StorageType StorageAreaImpl::storageType() const
{
return m_storageAreaMap->storageType();
}
unsigned StorageAreaImpl::length(ExceptionCode& ec, Frame* sourceFrame)
{
ec = 0;
if (!canAccessStorage(sourceFrame)) {
ec = SECURITY_ERR;
return 0;
}
if (disabledByPrivateBrowsingInFrame(sourceFrame))
return 0;
return m_storageAreaMap->length();
}
String StorageAreaImpl::key(unsigned index, ExceptionCode& ec, Frame* sourceFrame)
{
ec = 0;
if (!canAccessStorage(sourceFrame)) {
ec = SECURITY_ERR;
return String();
}
if (disabledByPrivateBrowsingInFrame(sourceFrame))
return String();
return m_storageAreaMap->key(index);
}
String StorageAreaImpl::getItem(const String& key, ExceptionCode& ec, Frame* sourceFrame)
{
ec = 0;
if (!canAccessStorage(sourceFrame)) {
ec = SECURITY_ERR;
return String();
}
if (disabledByPrivateBrowsingInFrame(sourceFrame))
return String();
return m_storageAreaMap->item(key);
}
void StorageAreaImpl::setItem(const String& key, const String& value, ExceptionCode& ec, Frame* sourceFrame)
{
ec = 0;
if (!canAccessStorage(sourceFrame)) {
ec = SECURITY_ERR;
return;
}
ASSERT(!value.isNull());
if (disabledByPrivateBrowsingInFrame(sourceFrame)) {
ec = QUOTA_EXCEEDED_ERR;
return;
}
bool quotaException;
m_storageAreaMap->setItem(sourceFrame, this, key, value, quotaException);
if (quotaException)
ec = QUOTA_EXCEEDED_ERR;
}
void StorageAreaImpl::removeItem(const String& key, ExceptionCode& ec, Frame* sourceFrame)
{
ec = 0;
if (!canAccessStorage(sourceFrame)) {
ec = SECURITY_ERR;
return;
}
if (disabledByPrivateBrowsingInFrame(sourceFrame))
return;
m_storageAreaMap->removeItem(sourceFrame, this, key);
}
void StorageAreaImpl::clear(ExceptionCode& ec, Frame* sourceFrame)
{
ec = 0;
if (!canAccessStorage(sourceFrame)) {
ec = SECURITY_ERR;
return;
}
if (disabledByPrivateBrowsingInFrame(sourceFrame))
return;
m_storageAreaMap->clear(sourceFrame, this);
}
bool StorageAreaImpl::contains(const String& key, ExceptionCode& ec, Frame* sourceFrame)
{
ec = 0;
if (!canAccessStorage(sourceFrame)) {
ec = SECURITY_ERR;
return false;
}
if (disabledByPrivateBrowsingInFrame(sourceFrame))
return false;
return m_storageAreaMap->contains(key);
}
bool StorageAreaImpl::canAccessStorage(Frame* frame)
{
return frame && frame->page();
}
size_t StorageAreaImpl::memoryBytesUsedByCache()
{
return 0;
}
void StorageAreaImpl::incrementAccessCount()
{
// Storage access is handled in the UI process, so there's nothing to do here.
}
void StorageAreaImpl::decrementAccessCount()
{
// Storage access is handled in the UI process, so there's nothing to do here.
}
void StorageAreaImpl::closeDatabaseIfIdle()
{
// FIXME: Implement this.
ASSERT_NOT_REACHED();
}
bool StorageAreaImpl::disabledByPrivateBrowsingInFrame(const Frame* sourceFrame) const
{
if (!sourceFrame->page()->settings()->privateBrowsingEnabled())
return false;
if (storageType() != LocalStorage)
return true;
return !SchemeRegistry::allowsLocalStorageAccessInPrivateBrowsing(sourceFrame->document()->securityOrigin()->protocol());
}
} // namespace WebKit