/* * 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 (Document* document = frame->document()) if (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