blob: a58acc0d14f84f747887acbe593de180690aad1b [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.
*/
WebInspector.StorageManager = function()
{
// FIXME: Convert this to a WebInspector.Object subclass, and call super().
// WebInspector.Object.call(this);
if (window.DOMStorageAgent)
DOMStorageAgent.enable();
if (window.DatabaseAgent)
DatabaseAgent.enable();
if (window.IndexedDBAgent)
IndexedDBAgent.enable();
WebInspector.Frame.addEventListener(WebInspector.Frame.Event.MainResourceDidChange, this._mainResourceDidChange, this);
WebInspector.notifications.addEventListener(WebInspector.Notification.ExtraDomainsActivated, this._extraDomainsActivated, this);
// COMPATIBILITY (iOS 6): DOMStorage was discovered via a DOMStorageObserver event. Now DOM Storage
// is added whenever a new securityOrigin is discovered. Check for DOMStorageAgent.getDOMStorageItems,
// which was renamed at the same time the change to start using securityOrigin was made.
if (window.DOMStorageAgent && DOMStorageAgent.getDOMStorageItems)
WebInspector.Frame.addEventListener(WebInspector.Frame.Event.SecurityOriginDidChange, this._securityOriginDidChange, this);
this.initialize();
};
WebInspector.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"
};
WebInspector.StorageManager.prototype = {
constructor: WebInspector.StorageManager,
__proto__: WebInspector.Object.prototype,
// Public
initialize: function()
{
this._domStorageObjects = [];
this._databaseObjects = [];
this._indexedDatabases = [];
this._cookieStorageObjects = {};
},
domStorageWasAdded: function(id, host, isLocalStorage)
{
var domStorage = new WebInspector.DOMStorageObject(id, host, isLocalStorage);
this._domStorageObjects.push(domStorage);
this.dispatchEventToListeners(WebInspector.StorageManager.Event.DOMStorageObjectWasAdded, {domStorage});
},
databaseWasAdded: function(id, host, name, version)
{
var database = new WebInspector.DatabaseObject(id, host, name, version);
this._databaseObjects.push(database);
this.dispatchEventToListeners(WebInspector.StorageManager.Event.DatabaseWasAdded, {database});
},
domStorageWasUpdated: function(id)
{
this.dispatchEventToListeners(WebInspector.StorageManager.Event.DOMStorageWasUpdated, id);
},
itemsCleared: function(storageId)
{
this._domStorageForIdentifier(storageId).itemsCleared(storageId);
},
itemRemoved: function(storageId, key)
{
this._domStorageForIdentifier(storageId).itemRemoved(key);
},
itemAdded: function(storageId, key, value)
{
this._domStorageForIdentifier(storageId).itemAdded(key, value);
},
itemUpdated: function(storageId, key, oldValue, value)
{
this._domStorageForIdentifier(storageId).itemUpdated(key, oldValue, value);
},
inspectDatabase: function(id)
{
var database = this._databaseForIdentifier(id);
console.assert(database);
if (!database)
return;
this.dispatchEventToListeners(WebInspector.StorageManager.Event.DatabaseWasInspected, {database});
},
inspectDOMStorage: function(id)
{
var domStorage = this._domStorageForIdentifier(id);
console.assert(domStorage);
if (!domStorage)
return;
this.dispatchEventToListeners(WebInspector.StorageManager.Event.DOMStorageObjectWasInspected, {domStorage});
},
// Protected
requestIndexedDatabaseData: function(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 = WebInspector.RemoteObject.fromPayload(entryPayload.primaryKey);
entry.key = WebInspector.RemoteObject.fromPayload(entryPayload.key);
entry.value = WebInspector.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);
},
// Private
_domStorageForIdentifier: function(id)
{
for (var storageObject of this._domStorageObjects) {
// The id is an object, so we need to compare the properties using Object.shallowEqual.
// COMPATIBILITY (iOS 6): The id was a string. Object.shallowEqual works for both.
if (Object.shallowEqual(storageObject.id, id))
return storageObject;
}
return null;
},
_mainResourceDidChange: function(event)
{
console.assert(event.target instanceof WebInspector.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(WebInspector.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 WebInspector.CookieStorageObject(host);
this.dispatchEventToListeners(WebInspector.StorageManager.Event.CookieStorageObjectWasAdded, {cookieStorage: this._cookieStorageObjects[host]});
},
_addDOMStorageIfNeeded: function(frame)
{
// 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: function(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 WebInspector.IndexedDatabase(databasePayload.name, securityOrigin, databasePayload.version, objectStores);
this._indexedDatabases.push(indexedDatabase);
this.dispatchEventToListeners(WebInspector.StorageManager.Event.IndexedDatabaseWasAdded, {indexedDatabase});
}
function processKeyPath(keyPathPayload)
{
switch (keyPathPayload.type) {
case "null":
return null;
case "string":
return keyPathPayload.string;
case "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 WebInspector.IndexedDatabaseObjectStore(objectStorePayload.name, keyPath, objectStorePayload.autoIncrement, indexes);
}
function processObjectStoreIndex(objectStoreIndexPayload)
{
var keyPath = processKeyPath(objectStoreIndexPayload.keyPath);
return new WebInspector.IndexedDatabaseObjectStoreIndex(objectStoreIndexPayload.name, keyPath, objectStoreIndexPayload.unique, objectStoreIndexPayload.multiEntry);
}
IndexedDBAgent.requestDatabaseNames(securityOrigin, processDatabaseNames.bind(this));
},
_securityOriginDidChange: function(event)
{
console.assert(event.target instanceof WebInspector.Frame);
this._addDOMStorageIfNeeded(event.target);
this._addIndexedDBDatabasesIfNeeded(event.target);
},
_databaseForIdentifier: function(id)
{
for (var i = 0; i < this._databaseObjects.length; ++i) {
if (this._databaseObjects[i].id === id)
return this._databaseObjects[i];
}
return null;
},
_extraDomainsActivated: function(event)
{
if (event.data.domains.contains("DOMStorage") && window.DOMStorageAgent && DOMStorageAgent.getDOMStorageItems)
WebInspector.Frame.addEventListener(WebInspector.Frame.Event.SecurityOriginDidChange, this._securityOriginDidChange, this);
}
};