| /* |
| * 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 "LocalStorage.h" |
| |
| #include "CString.h" |
| #include "EventNames.h" |
| #include "FileSystem.h" |
| #include "Frame.h" |
| #include "FrameTree.h" |
| #include "LocalStorageArea.h" |
| #include "Page.h" |
| #include "PageGroup.h" |
| #include "StorageArea.h" |
| |
| namespace WebCore { |
| |
| typedef HashMap<String, LocalStorage*> LocalStorageMap; |
| |
| static LocalStorageMap& localStorageMap() |
| { |
| static LocalStorageMap localStorageMap; |
| return localStorageMap; |
| } |
| |
| PassRefPtr<LocalStorage> LocalStorage::localStorage(const String& path) |
| { |
| const String lookupPath = path.isNull() ? String("") : path; |
| LocalStorageMap::iterator it = localStorageMap().find(lookupPath); |
| if (it == localStorageMap().end()) { |
| RefPtr<LocalStorage> localStorage = adoptRef(new LocalStorage(lookupPath)); |
| localStorageMap().set(lookupPath, localStorage.get()); |
| return localStorage.release(); |
| } |
| |
| return it->second; |
| } |
| |
| LocalStorage::LocalStorage(const String& path) |
| : m_path(path.copy()) |
| { |
| // If the path is empty, we know we're never going to be using the thread for anything, so don't start it. |
| // In the future, we might also want to consider removing it from the DOM in that case - <rdar://problem/5960470> |
| if (path.isEmpty()) |
| return; |
| |
| m_thread = LocalStorageThread::create(); |
| m_thread->start(); |
| m_thread->scheduleImport(this); |
| } |
| |
| LocalStorage::~LocalStorage() |
| { |
| ASSERT(localStorageMap().get(m_path) == this); |
| localStorageMap().remove(m_path); |
| } |
| |
| PassRefPtr<StorageArea> LocalStorage::storageArea(Frame* sourceFrame, SecurityOrigin* origin) |
| { |
| ASSERT(isMainThread()); |
| |
| // FIXME: If the security origin in question has never had a storage area established, |
| // we need to ask a client call if establishing it is okay. If the client denies the request, |
| // this method will return null. |
| // The sourceFrame argument exists for the purpose of asking a client. |
| // To know if an area has previously been established, we need to wait until this LocalStorage |
| // object has finished it's AreaImport task. |
| |
| |
| // FIXME: If the storage area is being established for the first time here, we need to |
| // sync its existance and quota out to disk via an task of type AreaSync |
| |
| RefPtr<LocalStorageArea> storageArea; |
| if (storageArea = m_storageAreaMap.get(origin)) |
| return storageArea.release(); |
| |
| storageArea = LocalStorageArea::create(origin, this); |
| m_storageAreaMap.set(origin, storageArea); |
| return storageArea.release(); |
| } |
| |
| String LocalStorage::fullDatabaseFilename(SecurityOrigin* origin) |
| { |
| // FIXME: Once we actually track origin/quota entries to see which origins have local storage established, |
| // we will return an empty path name if the origin isn't allowed to have LocalStorage. |
| // We'll need to wait here until the AreaImport task to complete before making that decision. |
| |
| if (m_path.isEmpty()) |
| return String(); |
| |
| ASSERT(origin); |
| if (!origin) |
| return String(); |
| |
| if (!makeAllDirectories(m_path)) { |
| LOG_ERROR("Unabled to create LocalStorage database path %s", m_path.utf8().data()); |
| return String(); |
| } |
| |
| return pathByAppendingComponent(m_path, origin->databaseIdentifier() + ".localstorage"); |
| } |
| |
| void LocalStorage::performImport() |
| { |
| ASSERT(!isMainThread()); |
| |
| // FIXME: Import all known local storage origins here along with their quotas |
| } |
| |
| void LocalStorage::performSync() |
| { |
| ASSERT(!isMainThread()); |
| |
| // FIXME: Write out new origins and quotas here |
| } |
| |
| void LocalStorage::close() |
| { |
| ASSERT(isMainThread()); |
| |
| LocalStorageAreaMap::iterator end = m_storageAreaMap.end(); |
| for (LocalStorageAreaMap::iterator it = m_storageAreaMap.begin(); it != end; ++it) |
| it->second->scheduleFinalSync(); |
| |
| if (m_thread) { |
| m_thread->terminate(); |
| m_thread = 0; |
| } |
| } |
| |
| bool LocalStorage::scheduleImport(PassRefPtr<LocalStorageArea> area) |
| { |
| ASSERT(isMainThread()); |
| |
| if (m_thread) |
| m_thread->scheduleImport(area); |
| |
| return m_thread; |
| } |
| |
| void LocalStorage::scheduleSync(PassRefPtr<LocalStorageArea> area) |
| { |
| ASSERT(isMainThread()); |
| if (m_thread) |
| m_thread->scheduleSync(area); |
| } |
| |
| } // namespace WebCore |