⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 localstoragearea.cpp

📁 linux下开源浏览器WebKit的源码,市面上的很多商用浏览器都是移植自WebKit
💻 CPP
字号:
/* * Copyright (C) 2008 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. ``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 * 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 "LocalStorageArea.h"#include "CString.h"#include "EventNames.h"#include "Frame.h"#include "FrameTree.h"#include "LocalStorage.h"#include "LocalStorageTask.h"#include "LocalStorageThread.h"#include "Page.h"#include "PageGroup.h"#include "PlatformString.h"#include "SecurityOrigin.h"#include "SQLiteStatement.h"namespace WebCore {// If the LocalStorageArea undergoes rapid changes, don't sync each change to disk.// Instead, queue up a batch of items to sync and actually do the sync at the following interval.static const double LocalStorageSyncInterval = 1.0;LocalStorageArea::LocalStorageArea(SecurityOrigin* origin, LocalStorage* localStorage)    : StorageArea(origin)    , m_syncTimer(this, &LocalStorageArea::syncTimerFired)    , m_itemsCleared(false)    , m_finalSyncScheduled(false)    , m_localStorage(localStorage)    , m_clearItemsWhileSyncing(false)    , m_syncScheduled(false)    , m_importComplete(false){    ASSERT(m_localStorage);        if (!m_localStorage->scheduleImport(this))        m_importComplete = true;}LocalStorageArea::~LocalStorageArea(){    ASSERT(!m_syncTimer.isActive());}void LocalStorageArea::scheduleFinalSync(){    m_syncTimer.stop();    syncTimerFired(&m_syncTimer);    m_finalSyncScheduled = true;}unsigned LocalStorageArea::length() const{    ASSERT(isMainThread());    if (m_importComplete)        return internalLength();    MutexLocker locker(m_importLock);    if (m_importComplete)        return internalLength();    while (!m_importComplete)        m_importCondition.wait(m_importLock);    ASSERT(m_importComplete);        return internalLength();}String LocalStorageArea::key(unsigned index, ExceptionCode& ec) const{    ASSERT(isMainThread());    if (m_importComplete)        return internalKey(index, ec);    MutexLocker locker(m_importLock);    if (m_importComplete)        return internalKey(index, ec);    while (!m_importComplete)        m_importCondition.wait(m_importLock);    ASSERT(m_importComplete);    return internalKey(index, ec);}String LocalStorageArea::getItem(const String& key) const{    ASSERT(isMainThread());    if (m_importComplete)        return internalGetItem(key);    MutexLocker locker(m_importLock);    if (m_importComplete)        return internalGetItem(key);    String item = internalGetItem(key);    if (!item.isNull())        return item;    while (!m_importComplete)        m_importCondition.wait(m_importLock);    ASSERT(m_importComplete);    return internalGetItem(key);}void LocalStorageArea::setItem(const String& key, const String& value, ExceptionCode& ec, Frame* frame){    ASSERT(isMainThread());    if (m_importComplete) {        internalSetItem(key, value, ec, frame);        return;    }    MutexLocker locker(m_importLock);    internalSetItem(key, value, ec, frame);}void LocalStorageArea::removeItem(const String& key, Frame* frame){        ASSERT(isMainThread());    if (m_importComplete) {        internalRemoveItem(key, frame);        return;    }    MutexLocker locker(m_importLock);    internalRemoveItem(key, frame);}bool LocalStorageArea::contains(const String& key) const{    ASSERT(isMainThread());    if (m_importComplete)        return internalContains(key);    MutexLocker locker(m_importLock);    if (m_importComplete)        return internalContains(key);    bool contained = internalContains(key);    if (contained)        return true;    while (!m_importComplete)        m_importCondition.wait(m_importLock);    ASSERT(m_importComplete);    return internalContains(key);}void LocalStorageArea::itemChanged(const String& key, const String& oldValue, const String& newValue, Frame* sourceFrame){    ASSERT(isMainThread());    scheduleItemForSync(key, newValue);    dispatchStorageEvent(key, oldValue, newValue, sourceFrame);}void LocalStorageArea::itemRemoved(const String& key, const String& oldValue, Frame* sourceFrame){    ASSERT(isMainThread());    scheduleItemForSync(key, String());    dispatchStorageEvent(key, oldValue, String(), sourceFrame);}void LocalStorageArea::areaCleared(Frame* sourceFrame){    ASSERT(isMainThread());    scheduleClear();    dispatchStorageEvent(String(), String(), String(), sourceFrame);}void LocalStorageArea::dispatchStorageEvent(const String& key, const String& oldValue, const String& newValue, Frame* sourceFrame){    ASSERT(isMainThread());    Page* page = sourceFrame->page();    if (!page)        return;    // Need to copy all relevant frames from every page to a vector, since sending the event to one frame might mutate the frame tree    // of any given page in the group, or mutate the page group itself    Vector<RefPtr<Frame> > frames;    const HashSet<Page*>& pages = page->group().pages();        HashSet<Page*>::const_iterator end = pages.end();    for (HashSet<Page*>::const_iterator it = pages.begin(); it != end; ++it) {        for (Frame* frame = (*it)->mainFrame(); frame; frame = frame->tree()->traverseNext()) {            if (frame->document()->securityOrigin()->equal(securityOrigin()))                frames.append(frame);        }    }    for (unsigned i = 0; i < frames.size(); ++i) {        if (HTMLElement* body = frames[i]->document()->body())            body->dispatchStorageEvent(eventNames().storageEvent, key, oldValue, newValue, sourceFrame);            }}void LocalStorageArea::scheduleItemForSync(const String& key, const String& value){    ASSERT(isMainThread());    ASSERT(!m_finalSyncScheduled);    m_changedItems.set(key, value);    if (!m_syncTimer.isActive())        m_syncTimer.startOneShot(LocalStorageSyncInterval);}void LocalStorageArea::scheduleClear(){    ASSERT(isMainThread());    ASSERT(!m_finalSyncScheduled);    m_changedItems.clear();    m_itemsCleared = true;    if (!m_syncTimer.isActive())        m_syncTimer.startOneShot(LocalStorageSyncInterval);}void LocalStorageArea::syncTimerFired(Timer<LocalStorageArea>*){    ASSERT(isMainThread());    HashMap<String, String>::iterator it = m_changedItems.begin();    HashMap<String, String>::iterator end = m_changedItems.end();        {        MutexLocker locker(m_syncLock);        if (m_itemsCleared) {            m_itemsPendingSync.clear();            m_clearItemsWhileSyncing = true;            m_itemsCleared = false;        }        for (; it != end; ++it)            m_itemsPendingSync.set(it->first.copy(), it->second.copy());        if (!m_syncScheduled) {            m_syncScheduled = true;            m_localStorage->scheduleSync(this);        }    }    m_changedItems.clear();}void LocalStorageArea::performImport(){    ASSERT(!isMainThread());    ASSERT(!m_database.isOpen());    String databaseFilename = m_localStorage->fullDatabaseFilename(securityOrigin());        if (databaseFilename.isEmpty()) {        LOG_ERROR("Filename for local storage database is empty - cannot open for persistent storage");        markImported();        return;    }    if (!m_database.open(databaseFilename)) {        LOG_ERROR("Failed to open database file %s for local storage", databaseFilename.utf8().data());        markImported();        return;    }    if (!m_database.executeCommand("CREATE TABLE IF NOT EXISTS ItemTable (key TEXT UNIQUE ON CONFLICT REPLACE, value TEXT NOT NULL ON CONFLICT FAIL)")) {        LOG_ERROR("Failed to create table ItemTable for local storage");        markImported();        return;    }        SQLiteStatement query(m_database, "SELECT key, value FROM ItemTable");    if (query.prepare() != SQLResultOk) {        LOG_ERROR("Unable to select items from ItemTable for local storage");        markImported();        return;    }        HashMap<String, String> itemMap;    int result = query.step();    while (result == SQLResultRow) {        itemMap.set(query.getColumnText(0), query.getColumnText(1));        result = query.step();    }    if (result != SQLResultDone) {        LOG_ERROR("Error reading items from ItemTable for local storage");        markImported();        return;    }    MutexLocker locker(m_importLock);        HashMap<String, String>::iterator it = itemMap.begin();    HashMap<String, String>::iterator end = itemMap.end();        for (; it != end; ++it)        importItem(it->first, it->second);        m_importComplete = true;    m_importCondition.signal();}void LocalStorageArea::markImported(){    ASSERT(!isMainThread());    MutexLocker locker(m_importLock);    m_importComplete = true;    m_importCondition.signal();}void LocalStorageArea::performSync(){    ASSERT(!isMainThread());    if (!m_database.isOpen())        return;    HashMap<String, String> items;    bool clearFirst = false;    {        MutexLocker locker(m_syncLock);        m_itemsPendingSync.swap(items);        clearFirst = m_clearItemsWhileSyncing;        m_clearItemsWhileSyncing = false;        m_syncScheduled = false;    }    // If the clear flag is marked, then we clear all items out before we write any new ones in    if (clearFirst) {        SQLiteStatement clear(m_database, "DELETE FROM ItemTable");        if (clear.prepare() != SQLResultOk) {            LOG_ERROR("Failed to prepare clear statement - cannot write to local storage database");            return;        }                int result = clear.step();        if (result != SQLResultDone) {            LOG_ERROR("Failed to clear all items in the local storage database - %i", result);            return;        }    }    SQLiteStatement insert(m_database, "INSERT INTO ItemTable VALUES (?, ?)");    if (insert.prepare() != SQLResultOk) {        LOG_ERROR("Failed to prepare insert statement - cannot write to local storage database");        return;    }    SQLiteStatement remove(m_database, "DELETE FROM ItemTable WHERE key=?");    if (remove.prepare() != SQLResultOk) {        LOG_ERROR("Failed to prepare delete statement - cannot write to local storage database");        return;    }    HashMap<String, String>::iterator end = items.end();    for (HashMap<String, String>::iterator it = items.begin(); it != end; ++it) {        // Based on the null-ness of the second argument, decide whether this is an insert or a delete        SQLiteStatement& query = it->second.isNull() ? remove : insert;                query.bindText(1, it->first);        // If the second argument is non-null, we're doing an insert, so bind it as the value.         if (!it->second.isNull())            query.bindText(2, it->second);        int result = query.step();        if (result != SQLResultDone) {            LOG_ERROR("Failed to update item in the local storage database - %i", result);            break;        }        query.reset();    }}} // namespace WebCore

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -