blob: 591d74db0a7ab3c024b11e7cde750b470f945143 [file] [log] [blame]
/*
* Copyright (C) 2016-2018 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. 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.TargetManager = class TargetManager extends WI.Object
{
constructor()
{
super();
this._targets = new Map;
this._cachedTargetsList = null;
this._seenPageTarget = false;
this._transitionTimeoutIdentifier = undefined;
}
// Public
get targets()
{
if (!this._cachedTargetsList)
this._cachedTargetsList = Array.from(this._targets.values()).filter((target) => !(target instanceof WI.MultiplexingBackendTarget));
return this._cachedTargetsList;
}
get allTargets()
{
return Array.from(this._targets.values());
}
targetForIdentifier(targetId)
{
if (!targetId)
return null;
for (let target of this._targets.values()) {
if (target.identifier === targetId)
return target;
}
return null;
}
targetCreated(targetInfo)
{
// Called from WI.TargetObserver.
// FIXME: Eliminate this once the local inspector is configured to use
// the Multiplexing code path. Then we can perform this immediately
// in `WI.loaded` if a TargetAgent exists.
if (this._targets.size === 0)
this.createMultiplexingBackendTarget(targetInfo);
let connection = new InspectorBackend.TargetConnection(targetInfo.targetId);
let target = this._createTarget(targetInfo, connection);
this._checkAndHandlePageTargetTransition(target);
target.initialize();
this.addTarget(target);
}
targetDestroyed(targetId)
{
// Called from WI.TargetObserver.
let target = this._targets.get(targetId);
this._checkAndHandlePageTargetTermination(target);
this.removeTarget(target);
}
dispatchMessageFromTarget(targetId, message)
{
// Called from WI.TargetObserver.
let target = this._targets.get(targetId);
console.assert(target);
if (!target)
return;
target.connection.dispatch(message);
}
addTarget(target)
{
console.assert(target);
console.assert(!this._targets.has(target.identifier));
this._cachedTargetsList = null;
this._targets.set(target.identifier, target);
this.dispatchEventToListeners(WI.TargetManager.Event.TargetAdded, {target});
}
removeTarget(target)
{
console.assert(target);
console.assert(target !== WI.mainTarget);
this._cachedTargetsList = null;
this._targets.delete(target.identifier);
this.dispatchEventToListeners(WI.TargetManager.Event.TargetRemoved, {target});
}
createMultiplexingBackendTarget(targetInfo)
{
let target = new WI.MultiplexingBackendTarget;
target.initialize();
WI.initializeBackendTarget(target);
// Add the target without dispatching an event.
this._targets.set(target.identifier, target);
}
createDirectBackendTarget()
{
let target = new WI.DirectBackendTarget;
target.initialize();
WI.initializeBackendTarget(target);
if (WI.sharedApp.debuggableType === WI.DebuggableType.Web)
WI.initializePageTarget(target);
this.addTarget(target);
}
// Private
_createTarget(targetInfo, connection)
{
let {targetId, type} = targetInfo;
switch (type) {
case TargetAgent.TargetInfoType.JavaScript:
return new WI.JavaScriptContextTarget(targetId, WI.UIString("JavaScript Context"), connection);
case TargetAgent.TargetInfoType.Page:
return new WI.PageTarget(targetId, WI.UIString("Page"), connection);
case TargetAgent.TargetInfoType.Worker:
return new WI.WorkerTarget(targetId, WI.UIString("Worker"), connection);
case TargetAgent.TargetInfoType.ServiceWorker:
return new WI.WorkerTarget(targetId, WI.UIString("ServiceWorker"), connection);
}
throw "Unknown Target type: " + type;
}
_checkAndHandlePageTargetTransition(target)
{
if (target.type !== WI.Target.Type.Page)
return;
// First page target.
if (!WI.pageTarget && !this._seenPageTarget) {
this._seenPageTarget = true;
WI.initializePageTarget(target);
return;
}
// Transitioning page target.
WI.transitionPageTarget(target);
}
_checkAndHandlePageTargetTermination(target)
{
if (target.type !== WI.Target.Type.Page)
return;
console.assert(target === WI.pageTarget);
console.assert(this._seenPageTarget);
// Terminating the page target.
WI.terminatePageTarget(target);
// Ensure we transition in a reasonable amount of time, otherwise close.
const timeToTransition = 2000;
clearTimeout(this._transitionTimeoutIdentifier);
this._transitionTimeoutIdentifier = setTimeout(() => {
this._transitionTimeoutIdentifier = undefined;
if (WI.pageTarget)
return;
if (WI.isEngineeringBuild)
throw new Error("Error: No new pageTarget some time after last page target was terminated. Failed transition?");
WI.close();
}, timeToTransition);
}
};
WI.TargetManager.Event = {
TargetAdded: Symbol("target-manager-target-added"),
TargetRemoved: Symbol("target-manager-target-removed"),
};