chrome-devtools-frontend
Version:
Chrome DevTools UI
287 lines (238 loc) • 9.95 kB
text/typescript
// Copyright 2021 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
/*
* Copyright (C) 2008 Nokia 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.
* 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of
* its contributors may be used to endorse or promote products derived
* from this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED "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 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.
*/
import * as Common from '../../core/common/common.js';
import * as SDK from '../../core/sdk/sdk.js';
import type * as ProtocolProxyApi from '../../generated/protocol-proxy-api.js';
import type * as Protocol from '../../generated/protocol.js';
export class DOMStorage extends Common.ObjectWrapper.ObjectWrapper<DOMStorage.EventTypes> {
private readonly model: DOMStorageModel;
private readonly storageKeyInternal: string;
private readonly isLocalStorageInternal: boolean;
constructor(model: DOMStorageModel, storageKey: string, isLocalStorage: boolean) {
super();
this.model = model;
this.storageKeyInternal = storageKey;
this.isLocalStorageInternal = isLocalStorage;
}
static storageId(storageKey: string, isLocalStorage: boolean): Protocol.DOMStorage.StorageId {
return {storageKey, isLocalStorage};
}
get id(): Protocol.DOMStorage.StorageId {
return DOMStorage.storageId(this.storageKeyInternal, this.isLocalStorageInternal);
}
get storageKey(): string|null {
return this.storageKeyInternal;
}
get isLocalStorage(): boolean {
return this.isLocalStorageInternal;
}
getItems(): Promise<Protocol.DOMStorage.Item[]|null> {
return this.model.agent.invoke_getDOMStorageItems({storageId: this.id}).then(({entries}) => entries);
}
setItem(key: string, value: string): void {
void this.model.agent.invoke_setDOMStorageItem({storageId: this.id, key, value});
}
removeItem(key: string): void {
void this.model.agent.invoke_removeDOMStorageItem({storageId: this.id, key});
}
clear(): void {
void this.model.agent.invoke_clear({storageId: this.id});
}
}
export namespace DOMStorage {
export const enum Events {
DOM_STORAGE_ITEMS_CLEARED = 'DOMStorageItemsCleared',
DOM_STORAGE_ITEM_REMOVED = 'DOMStorageItemRemoved',
DOM_STORAGE_ITEM_ADDED = 'DOMStorageItemAdded',
DOM_STORAGE_ITEM_UPDATED = 'DOMStorageItemUpdated',
}
export interface DOMStorageItemRemovedEvent {
key: string;
}
export interface DOMStorageItemAddedEvent {
key: string;
value: string;
}
export interface DOMStorageItemUpdatedEvent {
key: string;
oldValue: string;
value: string;
}
export interface EventTypes {
[Events.DOM_STORAGE_ITEMS_CLEARED]: void;
[Events.DOM_STORAGE_ITEM_REMOVED]: DOMStorageItemRemovedEvent;
[Events.DOM_STORAGE_ITEM_ADDED]: DOMStorageItemAddedEvent;
[Events.DOM_STORAGE_ITEM_UPDATED]: DOMStorageItemUpdatedEvent;
}
}
export class DOMStorageModel extends SDK.SDKModel.SDKModel<EventTypes> {
private readonly storageKeyManagerInternal: SDK.StorageKeyManager.StorageKeyManager|null;
private storagesInternal: Record<string, DOMStorage>;
readonly agent: ProtocolProxyApi.DOMStorageApi;
private enabled?: boolean;
constructor(target: SDK.Target.Target) {
super(target);
this.storageKeyManagerInternal = target.model(SDK.StorageKeyManager.StorageKeyManager);
this.storagesInternal = {};
this.agent = target.domstorageAgent();
}
enable(): void {
if (this.enabled) {
return;
}
this.target().registerDOMStorageDispatcher(new DOMStorageDispatcher(this));
if (this.storageKeyManagerInternal) {
this.storageKeyManagerInternal.addEventListener(
SDK.StorageKeyManager.Events.STORAGE_KEY_ADDED, this.storageKeyAdded, this);
this.storageKeyManagerInternal.addEventListener(
SDK.StorageKeyManager.Events.STORAGE_KEY_REMOVED, this.storageKeyRemoved, this);
for (const storageKey of this.storageKeyManagerInternal.storageKeys()) {
this.addStorageKey(storageKey);
}
}
void this.agent.invoke_enable();
this.enabled = true;
}
clearForStorageKey(storageKey: string): void {
if (!this.enabled) {
return;
}
for (const isLocal of [true, false]) {
const key = this.storageKey(storageKey, isLocal);
const storage = this.storagesInternal[key];
if (!storage) {
return;
}
storage.clear();
}
this.removeStorageKey(storageKey);
this.addStorageKey(storageKey);
}
private storageKeyAdded(event: Common.EventTarget.EventTargetEvent<string>): void {
this.addStorageKey(event.data);
}
private addStorageKey(storageKey: string): void {
for (const isLocal of [true, false]) {
const key = this.storageKey(storageKey, isLocal);
console.assert(!this.storagesInternal[key]);
const storage = new DOMStorage(this, storageKey, isLocal);
this.storagesInternal[key] = storage;
this.dispatchEventToListeners(Events.DOM_STORAGE_ADDED, storage);
}
}
private storageKeyRemoved(event: Common.EventTarget.EventTargetEvent<string>): void {
this.removeStorageKey(event.data);
}
private removeStorageKey(storageKey: string): void {
for (const isLocal of [true, false]) {
const key = this.storageKey(storageKey, isLocal);
const storage = this.storagesInternal[key];
if (!storage) {
continue;
}
delete this.storagesInternal[key];
this.dispatchEventToListeners(Events.DOM_STORAGE_REMOVED, storage);
}
}
private storageKey(storageKey: string, isLocalStorage: boolean): string {
return JSON.stringify(DOMStorage.storageId(storageKey, isLocalStorage));
}
domStorageItemsCleared(storageId: Protocol.DOMStorage.StorageId): void {
const domStorage = this.storageForId(storageId);
if (!domStorage) {
return;
}
domStorage.dispatchEventToListeners(DOMStorage.Events.DOM_STORAGE_ITEMS_CLEARED);
}
domStorageItemRemoved(storageId: Protocol.DOMStorage.StorageId, key: string): void {
const domStorage = this.storageForId(storageId);
if (!domStorage) {
return;
}
const eventData = {key};
domStorage.dispatchEventToListeners(DOMStorage.Events.DOM_STORAGE_ITEM_REMOVED, eventData);
}
domStorageItemAdded(storageId: Protocol.DOMStorage.StorageId, key: string, value: string): void {
const domStorage = this.storageForId(storageId);
if (!domStorage) {
return;
}
const eventData = {key, value};
domStorage.dispatchEventToListeners(DOMStorage.Events.DOM_STORAGE_ITEM_ADDED, eventData);
}
domStorageItemUpdated(storageId: Protocol.DOMStorage.StorageId, key: string, oldValue: string, value: string): void {
const domStorage = this.storageForId(storageId);
if (!domStorage) {
return;
}
const eventData = {key, oldValue, value};
domStorage.dispatchEventToListeners(DOMStorage.Events.DOM_STORAGE_ITEM_UPDATED, eventData);
}
storageForId(storageId: Protocol.DOMStorage.StorageId): DOMStorage {
console.assert(Boolean(storageId.storageKey));
return this.storagesInternal[this.storageKey(storageId.storageKey || '', storageId.isLocalStorage)];
}
storages(): DOMStorage[] {
const result = [];
for (const id in this.storagesInternal) {
result.push(this.storagesInternal[id]);
}
return result;
}
}
SDK.SDKModel.SDKModel.register(DOMStorageModel, {capabilities: SDK.Target.Capability.DOM, autostart: false});
export const enum Events {
DOM_STORAGE_ADDED = 'DOMStorageAdded',
DOM_STORAGE_REMOVED = 'DOMStorageRemoved',
}
export interface EventTypes {
[Events.DOM_STORAGE_ADDED]: DOMStorage;
[Events.DOM_STORAGE_REMOVED]: DOMStorage;
}
export class DOMStorageDispatcher implements ProtocolProxyApi.DOMStorageDispatcher {
private readonly model: DOMStorageModel;
constructor(model: DOMStorageModel) {
this.model = model;
}
domStorageItemsCleared({storageId}: Protocol.DOMStorage.DomStorageItemsClearedEvent): void {
this.model.domStorageItemsCleared(storageId);
}
domStorageItemRemoved({storageId, key}: Protocol.DOMStorage.DomStorageItemRemovedEvent): void {
this.model.domStorageItemRemoved(storageId, key);
}
domStorageItemAdded({storageId, key, newValue}: Protocol.DOMStorage.DomStorageItemAddedEvent): void {
this.model.domStorageItemAdded(storageId, key, newValue);
}
domStorageItemUpdated({storageId, key, oldValue, newValue}: Protocol.DOMStorage.DomStorageItemUpdatedEvent): void {
this.model.domStorageItemUpdated(storageId, key, oldValue, newValue);
}
}