blob: d3e68cb5d01429bffb1762e3210a763f2c3935ac [file] [log] [blame]
/*
* Copyright (C) 2013 Apple Inc. All rights reserved.
* Copyright (C) 2013 Samsung Electronics. 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.
*/
WI.StorageManager = class StorageManager extends WI.Object
{
constructor()
{
super();
if (window.DOMStorageAgent)
DOMStorageAgent.enable();
if (window.DatabaseAgent)
DatabaseAgent.enable();
if (window.IndexedDBAgent)
IndexedDBAgent.enable();
WI.Frame.addEventListener(WI.Frame.Event.MainResourceDidChange, this._mainResourceDidChange, this);
WI.Frame.addEventListener(WI.Frame.Event.SecurityOriginDidChange, this._securityOriginDidChange, this);
this.initialize();
}
// Public
initialize()
{
this._domStorageObjects = [];
this._databaseObjects = [];
this._indexedDatabases = [];
this._cookieStorageObjects = {};
}
get domStorageObjects()
{
return this._domStorageObjects;
}
get databases()
{
return this._databaseObjects;
}
get indexedDatabases()
{
return this._indexedDatabases;
}
get cookieStorageObjects()
{
var cookieStorageObjects = [];
for (var host in this._cookieStorageObjects)
cookieStorageObjects.push(this._cookieStorageObjects[host]);
return cookieStorageObjects;
}
domStorageWasAdded(id, host, isLocalStorage)
{
var domStorage = new WI.DOMStorageObject(id, host, isLocalStorage);
this._domStorageObjects.push(domStorage);
this.dispatchEventToListeners(WI.StorageManager.Event.DOMStorageObjectWasAdded, {domStorage});
}
databaseWasAdded(id, host, name, version)
{
var database = new WI.DatabaseObject(id, host, name, version);
this._databaseObjects.push(database);
this.dispatchEventToListeners(WI.StorageManager.Event.DatabaseWasAdded, {database});
}
itemsCleared(storageId)
{
let domStorage = this._domStorageForIdentifier(storageId);
if (domStorage)
domStorage.itemsCleared(storageId);
}
itemRemoved(storageId, key)
{
let domStorage = this._domStorageForIdentifier(storageId);
if (domStorage)
domStorage.itemRemoved(key);
}
itemAdded(storageId, key, value)
{
let domStorage = this._domStorageForIdentifier(storageId);
if (domStorage)
domStorage.itemAdded(key, value);
}
itemUpdated(storageId, key, oldValue, value)
{
let domStorage = this._domStorageForIdentifier(storageId);
if (domStorage)
domStorage.itemUpdated(key, oldValue, value);
}
inspectDatabase(id)
{
var database = this._databaseForIdentifier(id);
console.assert(database);
if (!database)
return;
this.dispatchEventToListeners(WI.StorageManager.Event.DatabaseWasInspected, {database});
}
inspectDOMStorage(id)
{
var domStorage = this._domStorageForIdentifier(id);
console.assert(domStorage);
if (!domStorage)
return;
this.dispatchEventToListeners(WI.StorageManager.Event.DOMStorageObjectWasInspected, {domStorage});
}
requestIndexedDatabaseData(objectStore, objectStoreIndex, startEntryIndex, maximumEntryCount, callback)
{
console.assert(window.IndexedDBAgent);
console.assert(objectStore);
console.assert(callback);
function processData(error, entryPayloads, moreAvailable)
{
if (error) {
callback(null, false);
return;
}
var entries = [];
for (var entryPayload of entryPayloads) {
var entry = {};
entry.primaryKey = WI.RemoteObject.fromPayload(entryPayload.primaryKey);
entry.key = WI.RemoteObject.fromPayload(entryPayload.key);
entry.value = WI.RemoteObject.fromPayload(entryPayload.value);
entries.push(entry);
}
callback(entries, moreAvailable);
}
var requestArguments = {
securityOrigin: objectStore.parentDatabase.securityOrigin,
databaseName: objectStore.parentDatabase.name,
objectStoreName: objectStore.name,
indexName: objectStoreIndex && objectStoreIndex.name || "",
skipCount: startEntryIndex || 0,
pageSize: maximumEntryCount || 100
};
IndexedDBAgent.requestData.invoke(requestArguments, processData);
}
clearObjectStore(objectStore)
{
let securityOrigin = objectStore.parentDatabase.securityOrigin;
let databaseName = objectStore.parentDatabase.name;
let objectStoreName = objectStore.name;
IndexedDBAgent.clearObjectStore(securityOrigin, databaseName, objectStoreName);
}
// Private
_domStorageForIdentifier(id)
{
for (var storageObject of this._domStorageObjects) {
// The id is an object, so we need to compare the properties using Object.shallowEqual.
if (Object.shallowEqual(storageObject.id, id))
return storageObject;
}
return null;
}
_mainResourceDidChange(event)
{
console.assert(event.target instanceof WI.Frame);
if (event.target.isMainFrame()) {
// If we are dealing with the main frame, we want to clear our list of objects, because we are navigating to a new page.
this.initialize();
this.dispatchEventToListeners(WI.StorageManager.Event.Cleared);
this._addDOMStorageIfNeeded(event.target);
this._addIndexedDBDatabasesIfNeeded(event.target);
}
// Add the host of the frame that changed the main resource to the list of hosts there could be cookies for.
var host = parseURL(event.target.url).host;
if (!host)
return;
if (this._cookieStorageObjects[host])
return;
this._cookieStorageObjects[host] = new WI.CookieStorageObject(host);
this.dispatchEventToListeners(WI.StorageManager.Event.CookieStorageObjectWasAdded, {cookieStorage: this._cookieStorageObjects[host]});
}
_addDOMStorageIfNeeded(frame)
{
if (!window.DOMStorageAgent)
return;
// Don't show storage if we don't have a security origin (about:blank).
if (!frame.securityOrigin || frame.securityOrigin === "://")
return;
// FIXME: Consider passing the other parts of the origin along to domStorageWasAdded.
var localStorageIdentifier = {securityOrigin: frame.securityOrigin, isLocalStorage: true};
if (!this._domStorageForIdentifier(localStorageIdentifier))
this.domStorageWasAdded(localStorageIdentifier, frame.mainResource.urlComponents.host, true);
var sessionStorageIdentifier = {securityOrigin: frame.securityOrigin, isLocalStorage: false};
if (!this._domStorageForIdentifier(sessionStorageIdentifier))
this.domStorageWasAdded(sessionStorageIdentifier, frame.mainResource.urlComponents.host, false);
}
_addIndexedDBDatabasesIfNeeded(frame)
{
if (!window.IndexedDBAgent)
return;
var securityOrigin = frame.securityOrigin;
// Don't show storage if we don't have a security origin (about:blank).
if (!securityOrigin || securityOrigin === "://")
return;
function processDatabaseNames(error, names)
{
if (error || !names)
return;
for (var name of names)
IndexedDBAgent.requestDatabase(securityOrigin, name, processDatabase.bind(this));
}
function processDatabase(error, databasePayload)
{
if (error || !databasePayload)
return;
var objectStores = databasePayload.objectStores.map(processObjectStore);
var indexedDatabase = new WI.IndexedDatabase(databasePayload.name, securityOrigin, databasePayload.version, objectStores);
this._indexedDatabases.push(indexedDatabase);
this.dispatchEventToListeners(WI.StorageManager.Event.IndexedDatabaseWasAdded, {indexedDatabase});
}
function processKeyPath(keyPathPayload)
{
switch (keyPathPayload.type) {
case IndexedDBAgent.KeyPathType.Null:
return null;
case IndexedDBAgent.KeyPathType.String:
return keyPathPayload.string;
case IndexedDBAgent.KeyPathType.Array:
return keyPathPayload.array;
default:
console.error("Unknown KeyPath type:", keyPathPayload.type);
return null;
}
}
function processObjectStore(objectStorePayload)
{
var keyPath = processKeyPath(objectStorePayload.keyPath);
var indexes = objectStorePayload.indexes.map(processObjectStoreIndex);
return new WI.IndexedDatabaseObjectStore(objectStorePayload.name, keyPath, objectStorePayload.autoIncrement, indexes);
}
function processObjectStoreIndex(objectStoreIndexPayload)
{
var keyPath = processKeyPath(objectStoreIndexPayload.keyPath);
return new WI.IndexedDatabaseObjectStoreIndex(objectStoreIndexPayload.name, keyPath, objectStoreIndexPayload.unique, objectStoreIndexPayload.multiEntry);
}
IndexedDBAgent.requestDatabaseNames(securityOrigin, processDatabaseNames.bind(this));
}
_securityOriginDidChange(event)
{
console.assert(event.target instanceof WI.Frame);
this._addDOMStorageIfNeeded(event.target);
this._addIndexedDBDatabasesIfNeeded(event.target);
}
_databaseForIdentifier(id)
{
for (var i = 0; i < this._databaseObjects.length; ++i) {
if (this._databaseObjects[i].id === id)
return this._databaseObjects[i];
}
return null;
}
};
WI.StorageManager.Event = {
CookieStorageObjectWasAdded: "storage-manager-cookie-storage-object-was-added",
DOMStorageObjectWasAdded: "storage-manager-dom-storage-object-was-added",
DOMStorageObjectWasInspected: "storage-dom-object-was-inspected",
DatabaseWasAdded: "storage-manager-database-was-added",
DatabaseWasInspected: "storage-object-was-inspected",
IndexedDatabaseWasAdded: "storage-manager-indexed-database-was-added",
Cleared: "storage-manager-cleared"
};