@adobe/uix-host 0.7.1-nightly.20230228 → 0.8.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/debug-host.d.ts.map +1 -1
- package/dist/host.d.ts +10 -10
- package/dist/host.d.ts.map +1 -1
- package/dist/index.js +74 -4
- package/dist/index.js.map +1 -1
- package/dist/port.d.ts +4 -10
- package/dist/port.d.ts.map +1 -1
- package/package.json +2 -2
- package/src/debug-host.ts +5 -7
- package/src/host.ts +15 -16
- package/src/port.ts +27 -29
package/dist/debug-host.d.ts.map
CHANGED
@@ -1 +1 @@
|
|
1
|
-
{"version":3,"file":"debug-host.d.ts","sourceRoot":"","sources":["../src/debug-host.ts"],"names":[],"mappings":"AAcA;;;GAGG;AACH,OAAO,EAEL,kBAAkB,EAClB,KAAK,EAEN,MAAM,iBAAiB,CAAC;
|
1
|
+
{"version":3,"file":"debug-host.d.ts","sourceRoot":"","sources":["../src/debug-host.ts"],"names":[],"mappings":"AAcA;;;GAGG;AACH,OAAO,EAEL,kBAAkB,EAClB,KAAK,EAEN,MAAM,iBAAiB,CAAC;AACzB,OAAO,KAAK,EAA0B,UAAU,EAAE,MAAM,WAAW,CAAC;AAEpE,wBAAgB,SAAS,CAAC,IAAI,EAAE,KAAK,CAAC,UAAU,CAAC,GAAG,kBAAkB,CAkDrE"}
|
package/dist/host.d.ts
CHANGED
@@ -1,27 +1,27 @@
|
|
1
|
-
import type { Extension,
|
1
|
+
import type { Extension, GuestEmitter, NamedEvent, GuestApis } from "@adobe/uix-core";
|
2
2
|
import type { CapabilitySpec } from "./port.js";
|
3
3
|
import { Emitter } from "@adobe/uix-core";
|
4
|
-
import { PortOptions } from "./port.js";
|
4
|
+
import { Port, PortOptions } from "./port.js";
|
5
5
|
/**
|
6
6
|
* Dictionary of {@link Port} objects by extension ID.
|
7
7
|
* @public
|
8
8
|
*/
|
9
|
-
export declare type PortMap = Map<string,
|
9
|
+
export declare type PortMap = Map<string, Port>;
|
10
10
|
/** @public */
|
11
11
|
export declare type HostEvent<Type extends string = string, Detail = Record<string, unknown>> = NamedEvent<Type, Detail & Record<string, unknown> & {
|
12
12
|
host: Host;
|
13
13
|
}>;
|
14
14
|
/** @public */
|
15
15
|
declare type HostGuestEvent<Type extends string> = HostEvent<`guest${Type}`, {
|
16
|
-
guest:
|
16
|
+
guest: Port;
|
17
17
|
}>;
|
18
18
|
/**
|
19
19
|
* All guests requested by host have been loaded and connected.
|
20
20
|
* @public
|
21
21
|
*/
|
22
22
|
export declare type HostEventLoadAllGuests = HostEvent<"loadallguests", {
|
23
|
-
failed:
|
24
|
-
loaded:
|
23
|
+
failed: Port[];
|
24
|
+
loaded: Port[];
|
25
25
|
}>;
|
26
26
|
/**
|
27
27
|
* Shared context has been set or updated; all guests receive this event too.
|
@@ -83,7 +83,7 @@ export interface HostConfig {
|
|
83
83
|
* Callback to use to filter the list returned from {@link Host.(getLoadedGuests:2)}
|
84
84
|
* @public
|
85
85
|
*/
|
86
|
-
declare type GuestFilter = (item:
|
86
|
+
declare type GuestFilter = (item: GuestEmitter) => boolean;
|
87
87
|
/**
|
88
88
|
* Manager object for connecting to {@link @adobe/uix-guest#GuestServer |
|
89
89
|
* GuestServers} and {@link @adobe/uix-guest#GuestUI | GuestUIs}, providing and
|
@@ -169,15 +169,15 @@ export declare class Host extends Emitter<HostEvents> {
|
|
169
169
|
/**
|
170
170
|
* Return all loaded guests.
|
171
171
|
*/
|
172
|
-
getLoadedGuests():
|
172
|
+
getLoadedGuests<T = unknown>(): Port<T>[];
|
173
173
|
/**
|
174
174
|
* Return loaded guests which satisfy the passed test function.
|
175
175
|
*/
|
176
|
-
getLoadedGuests(filter: GuestFilter):
|
176
|
+
getLoadedGuests<T = unknown>(filter: GuestFilter): Port<T>[];
|
177
177
|
/**
|
178
178
|
* Return loaded guests which expose the provided {@link CapabilitySpec}.
|
179
179
|
*/
|
180
|
-
getLoadedGuests<Apis extends GuestApis>(capabilities: CapabilitySpec<Apis>):
|
180
|
+
getLoadedGuests<Apis extends GuestApis>(capabilities: CapabilitySpec<Apis>): Port<GuestApis>[];
|
181
181
|
/**
|
182
182
|
* Set the object of shared values that all Guests can access via {@link @adobe/uix-guest#GuestServer.sharedContext}.
|
183
183
|
* This overwrites any previous object.
|
package/dist/host.d.ts.map
CHANGED
@@ -1 +1 @@
|
|
1
|
-
{"version":3,"file":"host.d.ts","sourceRoot":"","sources":["../src/host.ts"],"names":[],"mappings":"AAYA,OAAO,KAAK,EACV,SAAS,EACT,
|
1
|
+
{"version":3,"file":"host.d.ts","sourceRoot":"","sources":["../src/host.ts"],"names":[],"mappings":"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"}
|
package/dist/index.js
CHANGED
@@ -19,6 +19,8 @@ var mergeAttrValues = (...tokenLists) => {
|
|
19
19
|
// src/dom-utils/iframe-normalizers.ts
|
20
20
|
var makeSandboxAttrs = (...sandboxes) => mergeAttrValues(...sandboxes);
|
21
21
|
var requiredIframeProps = {
|
22
|
+
// must not require this until app builder supports CSP
|
23
|
+
// csp: "frame-ancestors 'self'",
|
22
24
|
"data-uix-guest": "true",
|
23
25
|
role: "presentation",
|
24
26
|
referrerPolicy: "strict-origin"
|
@@ -36,6 +38,8 @@ var defaultOptions = {
|
|
36
38
|
debug: false
|
37
39
|
};
|
38
40
|
var Port = class extends uixCore.Emitter {
|
41
|
+
// #endregion Properties (13)
|
42
|
+
// #region Constructors (1)
|
39
43
|
constructor(config) {
|
40
44
|
super(config.id);
|
41
45
|
this.hostApis = {};
|
@@ -63,9 +67,29 @@ var Port = class extends uixCore.Emitter {
|
|
63
67
|
return server && server.apis;
|
64
68
|
}
|
65
69
|
}
|
70
|
+
// #endregion Constructors (1)
|
71
|
+
// #region Public Methods (6)
|
72
|
+
/**
|
73
|
+
* Connect an iframe element which is displaying another page in the extension
|
74
|
+
* with the extension's bootstrap frame, so they can share context and events.
|
75
|
+
*/
|
66
76
|
attachUI(iframe) {
|
67
|
-
return this.attachFrame(iframe
|
77
|
+
return this.attachFrame(iframe, {
|
78
|
+
onIframeResize: (dimensions) => {
|
79
|
+
this.emit("guestresize", {
|
80
|
+
dimensions,
|
81
|
+
guestPort: this,
|
82
|
+
iframe
|
83
|
+
});
|
84
|
+
}
|
85
|
+
});
|
68
86
|
}
|
87
|
+
/**
|
88
|
+
* Returns true if the guest has registered methods matching the provided
|
89
|
+
* capability spec. A capability spec is simply an object whose properties are
|
90
|
+
* declared in an array of keys, description the names of the functions and
|
91
|
+
* methods that the Port will expose.
|
92
|
+
*/
|
69
93
|
hasCapabilities(requiredMethods) {
|
70
94
|
this.assertReady();
|
71
95
|
return Object.keys(requiredMethods).every((key) => {
|
@@ -79,9 +103,16 @@ var Port = class extends uixCore.Emitter {
|
|
79
103
|
);
|
80
104
|
});
|
81
105
|
}
|
106
|
+
/**
|
107
|
+
* True when al extensions have loaded.
|
108
|
+
*/
|
82
109
|
isReady() {
|
83
110
|
return this.isLoaded && !this.error;
|
84
111
|
}
|
112
|
+
/**
|
113
|
+
* Loads the extension. Returns a promise which resolves when the extension
|
114
|
+
* has loaded. The Host calls this method after retrieving extensions.
|
115
|
+
*/
|
85
116
|
async load() {
|
86
117
|
try {
|
87
118
|
if (!this.apis) {
|
@@ -94,10 +125,21 @@ var Port = class extends uixCore.Emitter {
|
|
94
125
|
throw e;
|
95
126
|
}
|
96
127
|
}
|
128
|
+
/**
|
129
|
+
* The host-side equivalent of {@link @adobe/uix-guest#register}. Pass a set
|
130
|
+
* of methods down to the guest as proxies.
|
131
|
+
* Merges at the first level, the API level. Overwrites a deeper levels.
|
132
|
+
*/
|
97
133
|
provide(apis) {
|
98
|
-
Object.
|
134
|
+
for (const [apiNamespace, methods] of Object.entries(apis)) {
|
135
|
+
this.hostApis[apiNamespace] = this.hostApis[apiNamespace] || {};
|
136
|
+
Object.assign(this.hostApis[apiNamespace], methods);
|
137
|
+
}
|
99
138
|
this.emit("hostprovide", { guestPort: this, apis });
|
100
139
|
}
|
140
|
+
/**
|
141
|
+
* Disconnect from the extension.
|
142
|
+
*/
|
101
143
|
async unload() {
|
102
144
|
if (this.guestServerFrame && this.guestServerFrame.parentElement) {
|
103
145
|
this.guestServerFrame.parentElement.removeChild(this.guestServerFrame);
|
@@ -105,6 +147,8 @@ var Port = class extends uixCore.Emitter {
|
|
105
147
|
}
|
106
148
|
this.emit("unload", { guestPort: this });
|
107
149
|
}
|
150
|
+
// #endregion Public Methods (6)
|
151
|
+
// #region Private Methods (6)
|
108
152
|
assert(condition, errorMessage) {
|
109
153
|
if (!condition) {
|
110
154
|
throw new Error(
|
@@ -115,8 +159,9 @@ var Port = class extends uixCore.Emitter {
|
|
115
159
|
assertReady() {
|
116
160
|
this.assert(this.isReady(), () => "Attempted to interact before loaded");
|
117
161
|
}
|
118
|
-
attachFrame(iframe) {
|
162
|
+
attachFrame(iframe, addedMethods = {}) {
|
119
163
|
normalizeIframe(iframe);
|
164
|
+
this.logger.log("attachFrame", iframe);
|
120
165
|
return uixCore.connectIframe(
|
121
166
|
iframe,
|
122
167
|
{
|
@@ -126,7 +171,8 @@ var Port = class extends uixCore.Emitter {
|
|
126
171
|
},
|
127
172
|
{
|
128
173
|
getSharedContext: () => this.sharedContext,
|
129
|
-
invokeHostMethod: (address) => this.invokeHostMethod(address)
|
174
|
+
invokeHostMethod: (address) => this.invokeHostMethod(address),
|
175
|
+
...addedMethods
|
130
176
|
}
|
131
177
|
);
|
132
178
|
}
|
@@ -203,6 +249,7 @@ var Port = class extends uixCore.Emitter {
|
|
203
249
|
...args
|
204
250
|
]);
|
205
251
|
}
|
252
|
+
// #endregion Private Methods (6)
|
206
253
|
};
|
207
254
|
function debugHost(host) {
|
208
255
|
const hostLogger = uixCore.debugEmitter(host, {
|
@@ -222,6 +269,8 @@ function debugHost(host) {
|
|
222
269
|
log2.info("received APIs", event2.detail.apis);
|
223
270
|
}).listen("beforecallhostmethod", (log2, event2) => {
|
224
271
|
log2.info(event2.detail);
|
272
|
+
}).listen("guestresize", (log2, event2) => {
|
273
|
+
log2.info(event2.detail);
|
225
274
|
}).listen("unload", (log2, event2) => {
|
226
275
|
log2.info(event2.detail);
|
227
276
|
log2.detach();
|
@@ -250,7 +299,14 @@ var passAllGuests = () => true;
|
|
250
299
|
var _Host = class extends uixCore.Emitter {
|
251
300
|
constructor(config) {
|
252
301
|
super(config.hostName);
|
302
|
+
/**
|
303
|
+
* `true` if any extension in {@link Host.guests} has created a {@link
|
304
|
+
* @adobe/uix-guest#GuestServer}, but the Guest has not yet loaded.
|
305
|
+
*/
|
253
306
|
this.loading = false;
|
307
|
+
/**
|
308
|
+
* A Map of of the loaded guests.
|
309
|
+
*/
|
254
310
|
this.guests = /* @__PURE__ */ new Map();
|
255
311
|
this.cachedCapabilityLists = /* @__PURE__ */ new WeakMap();
|
256
312
|
this.logger = uixCore.quietConsole;
|
@@ -290,6 +346,14 @@ var _Host = class extends uixCore.Emitter {
|
|
290
346
|
context: this.sharedContext
|
291
347
|
});
|
292
348
|
}
|
349
|
+
/**
|
350
|
+
* Load extension into host application from provided extension description.
|
351
|
+
* Returned promise resolves when all extensions are loaded and registered.
|
352
|
+
*
|
353
|
+
* @param extensions - List of extension descriptors. Normally, the Host should receive this value from an {@link ExtensionsProvider}.
|
354
|
+
* @param options - Custom options to be used as defaults for each {@link Port} object created for each guest.
|
355
|
+
* @returns Promise which resolves when all guests have been loaded.
|
356
|
+
*/
|
293
357
|
async load(extensions, options) {
|
294
358
|
this.runtimeContainer = this.runtimeContainer || this.createRuntimeContainer(window);
|
295
359
|
const failed = [];
|
@@ -304,6 +368,10 @@ var _Host = class extends uixCore.Emitter {
|
|
304
368
|
this.loading = false;
|
305
369
|
this.emit("loadallguests", { host: this, failed, loaded });
|
306
370
|
}
|
371
|
+
/**
|
372
|
+
* Unload all extensions and remove their frames/workers. Use this to unmount
|
373
|
+
* a UI or when switching to a different extensible UI.
|
374
|
+
*/
|
307
375
|
async unload() {
|
308
376
|
this.emit("beforeunload", { host: this });
|
309
377
|
await Promise.all([...this.guests.values()].map((guest) => guest.unload()));
|
@@ -400,6 +468,7 @@ async function fetchExtensionsFromRegistry(config) {
|
|
400
468
|
headers: {
|
401
469
|
Accept: "application/json",
|
402
470
|
Authorization: `${config.auth.schema} ${config.auth.imsToken}`,
|
471
|
+
// todo: check if auth schema needed (initial implementation was without it)
|
403
472
|
"X-Api-Key": config.apiKey
|
404
473
|
}
|
405
474
|
}
|
@@ -420,6 +489,7 @@ function extensionRegistryExtensionsProvider(config) {
|
|
420
489
|
}
|
421
490
|
return {
|
422
491
|
...a,
|
492
|
+
// todo: make safer way to extract href
|
423
493
|
[e.name]: e.endpoints[erEndpoint].view[0].href
|
424
494
|
};
|
425
495
|
}, {})
|
package/dist/index.js.map
CHANGED
@@ -1 +1 @@
|
|
1
|
-
{"version":3,"sources":["../src/host.ts","../src/port.ts","../src/dom-utils/attribute-normalizers.ts","../src/dom-utils/iframe-normalizers.ts","../src/debug-host.ts","../src/extensions-provider/extension-registry.ts","../src/extensions-provider/composition.ts","../src/extensions-provider/mute.ts"],"names":["Emitter","log","event","host","window"],"mappings":"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","sourcesContent":["/*\nCopyright 2022 Adobe. All rights reserved.\nThis file is licensed to you under the Apache License, Version 2.0 (the \"License\");\nyou may not use this file except in compliance with the License. You may obtain a copy\nof the License at http://www.apache.org/licenses/LICENSE-2.0\n\nUnless required by applicable law or agreed to in writing, software distributed under\nthe License is distributed on an \"AS IS\" BASIS, WITHOUT WARRANTIES OR REPRESENTATIONS\nOF ANY KIND, either express or implied. See the License for the specific language\ngoverning permissions and limitations under the License.\n*/\n\nimport type {\n Extension,\n GuestConnection,\n NamedEvent,\n Emits,\n GuestApis,\n} from \"@adobe/uix-core\";\nimport type { CapabilitySpec } from \"./port.js\";\nimport { Emitter, quietConsole } from \"@adobe/uix-core\";\nimport { Port, PortOptions } from \"./port.js\";\nimport { debugHost } from \"./debug-host.js\";\n\n/**\n * Dictionary of {@link Port} objects by extension ID.\n * @public\n */\nexport type PortMap = Map<string, GuestConnection>;\n\n/** @public */\nexport type HostEvent<\n Type extends string = string,\n Detail = Record<string, unknown>\n> = NamedEvent<Type, Detail & Record<string, unknown> & { host: Host }>;\n/** @public */\ntype HostGuestEvent<Type extends string> = HostEvent<\n `guest${Type}`,\n { guest: GuestConnection }\n>;\n\n/**\n * All guests requested by host have been loaded and connected.\n * @public\n */\nexport type HostEventLoadAllGuests = HostEvent<\n \"loadallguests\",\n { failed: GuestConnection[]; loaded: GuestConnection[] }\n>;\n\n/**\n * Shared context has been set or updated; all guests receive this event too.\n * @public\n */\n\nexport type HostEventContextChange = HostEvent<\n \"contextchange\",\n { context: SharedContextValues }\n>;\n\n/**\n * An error has occurred during loading or unloading of guests.\n * @public\n */\nexport type HostEventError = HostEvent<\"error\", { error: Error }>;\n\n/** @public */\nexport type HostEvents =\n | HostGuestEvent<\"beforeload\">\n | HostGuestEvent<\"load\">\n | HostEvent<\"beforeunload\">\n | HostEvent<\"unload\">\n | HostEventLoadAllGuests\n | HostEventContextChange\n | HostEventError;\n\n/** @public */\nexport type InstalledExtensions = Record<Extension[\"id\"], Extension[\"url\"]>;\n/** @public */\nexport type ExtensionsProvider = () => Promise<InstalledExtensions>;\n\n/**\n * Values for shared context. Must be a plain object, serializable to JSON.\n * @public\n */\nexport type SharedContextValues = Record<string, unknown>;\n\n/** @public */\nexport interface HostConfig {\n /**\n * Human-readable \"slug\" name of the extensible area--often an entire app.\n * This string serves as a namespace for extension points within the area.\n */\n hostName: string;\n /**\n * A DOM element _outside_ of the React root. This is necessary to preserve\n * the lifetime of the iframes which are running extension objects; if they\n * live inside the React root, then React could unexpectedly re-render the\n * iframe tags themselves at any time, causing a reload of the frame.\n */\n runtimeContainer?: HTMLElement;\n /**\n * Copiously log lifecycle events.\n */\n debug?: boolean;\n /**\n * Default options to use for every guest Port.\n *\n * If `config.debug` is true, then the guest options will have `debug: true`\n * unless `debug: false` is explicitly passed in `guestOptions`.\n */\n guestOptions?: PortOptions;\n /**\n * A read-only dictionary of values that the host will supply to all the\n * guests.\n */\n sharedContext?: SharedContextValues;\n}\n\n/**\n * Callback to use to filter the list returned from {@link Host.(getLoadedGuests:2)}\n * @public\n */\ntype GuestFilter = (item: GuestConnection) => boolean;\n\nconst passAllGuests = () => true;\n\n/**\n * Manager object for connecting to {@link @adobe/uix-guest#GuestServer |\n * GuestServers} and {@link @adobe/uix-guest#GuestUI | GuestUIs}, providing and\n * receiving their APIs, and providing them to the app for interacting with UI.\n *\n * @remarks\n * The Host object is the main connection manager for all UIX Guests.\n * Making an app extensible requires creating a Host object.\n *\n * The extensible app using the Hostis responsible for providing a list of\n * extension references to the Host object. Use {@link\n * createExtensionRegistryProvider} for that purpose. Once you have retrieved a\n * list of extensions available to the host app, pass it to {@link Host.load}.\n *\n * When a Host creates a Guest, it must create an `<iframe>` element to contain\n * the Guest's main {@link @adobe/uix-guest#GuestServer} runtime, which runs\n * invisibly in the background. To do this, the Host creates a hidden container\n * in the body of the document. It is a `<div>` element with the attribute\n * `data-uix-guest-container`. Loaded GuestServers will be injected into this\n * hidden element and removed as necessary. When {@link Host.unload} is called,\n * the Host removes the hidden container from the document after unloading.\n *\n * @public\n */\nexport class Host extends Emitter<HostEvents> {\n /**\n * {@inheritDoc HostEventLoadAllGuests}\n * @eventProperty\n */\n public loadallguests: HostEventLoadAllGuests;\n\n /**\n * One guest has loaded.\n * @eventProperty\n */\n public guestload: HostGuestEvent<\"load\">;\n\n /**\n * About to attempt to load and connect to a Guest.\n * @eventProperty\n */\n public guestbeforeload: HostGuestEvent<\"beforeload\">;\n\n /**\n * About to unload a guest and remove its {@link @adobe/uix-guest#GuestServer}\n * instance as well as all its {@link @adobe/uix-guest#GuestUI} instances.\n * @eventProperty\n */\n public guestbeforeunload: HostGuestEvent<\"beforeunload\">;\n\n /**\n * Unloaded a guest and removed its {@link @adobe/uix-guest#GuestServer}\n * instance as well as all its {@link @adobe/uix-guest#GuestUI} instances.\n * @eventProperty\n */\n public guestunload: HostGuestEvent<\"unload\">;\n\n /**\n * {@inheritDoc HostEventContextChange}\n * @eventProperty\n */\n public contextchange: HostEventContextChange;\n\n /**\n * {@inheritDoc HostEventError}\n * @eventProperty\n */\n public error: HostEventError;\n\n private static containerStyle = {\n position: \"fixed\",\n width: \"1px\",\n height: \"1px\",\n pointerEvents: \"none\",\n opacity: 0,\n top: 0,\n left: \"-1px\",\n };\n /**\n * Unique string identifying the Host object.\n */\n hostName: string;\n /**\n * `true` if any extension in {@link Host.guests} has created a {@link\n * @adobe/uix-guest#GuestServer}, but the Guest has not yet loaded.\n */\n loading = false;\n /**\n * A Map of of the loaded guests.\n */\n guests: PortMap = new Map();\n private cachedCapabilityLists: WeakMap<object, GuestConnection[]> =\n new WeakMap();\n private runtimeContainer: HTMLElement;\n private guestOptions: PortOptions;\n private logger: Console = quietConsole;\n private sharedContext: SharedContextValues;\n constructor(config: HostConfig) {\n super(config.hostName);\n const { guestOptions = {} } = config;\n this.guestOptions = {\n ...guestOptions,\n debug: guestOptions.debug === false ? false : !!config.debug,\n };\n this.hostName = config.hostName;\n this.sharedContext = config.sharedContext || {};\n this.runtimeContainer = config.runtimeContainer;\n if (config.debug) {\n this.logger = debugHost(this);\n }\n }\n /**\n * Return all loaded guests.\n */\n getLoadedGuests(): GuestConnection[];\n /**\n * Return loaded guests which satisfy the passed test function.\n */\n getLoadedGuests(filter: GuestFilter): GuestConnection[];\n /**\n * Return loaded guests which expose the provided {@link CapabilitySpec}.\n */\n getLoadedGuests<Apis extends GuestApis>(\n capabilities: CapabilitySpec<Apis>\n ): GuestConnection[];\n getLoadedGuests<Apis extends GuestApis = never>(\n filterOrCapabilities?: CapabilitySpec<Apis> | GuestFilter\n ): GuestConnection[] {\n if (typeof filterOrCapabilities === \"object\") {\n return this.getLoadedGuestsWith<Apis>(filterOrCapabilities);\n }\n const filter = filterOrCapabilities || passAllGuests;\n const result = [];\n for (const guest of this.guests.values()) {\n if (guest.isReady() && filter(guest)) {\n result.push(guest);\n }\n }\n return result;\n }\n /**\n * Set the object of shared values that all Guests can access via {@link @adobe/uix-guest#GuestServer.sharedContext}.\n * This overwrites any previous object.\n *\n * @example Exposes `authToken` to all Guests. Guests can call `this.sharedContext.get('authToken')` to retrieve this value.\n * ```javascript\n * host.shareContext({\n * authToken: '82ba19b'\n * });\n * ```\n *\n * @example Overwrites the previous sharedContext, deleting `authToken` and providing `secret` and `auth` instead.\n * ```javascript\n * host.shareContext({\n * secret: 'squirrel',\n * auth: false\n * });\n * ```\n */\n shareContext(context: SharedContextValues): void;\n /**\n * Update the object of shared values that all Guests can access via {@link\n * @adobe/uix-guest#GuestServer.sharedContext}. This method takes a callback\n * which receives the previous context and may return an entirely new context,\n * or new values merged with the old context.\n *\n * @remarks This callback pattern allows the shared context values to be\n * mutable while the internal context object references are immutable, which\n * is important for synchronizing. with guests.\n *\n * @example Overwrites a context object based on the previous one.\n * ```javascript\n * host.shareContext(oldContext => ({\n * counter: oldContext.counter + 1\n * }))\n * ```\n *\n * @example Updates a context while preserving other existing values.\n * ```javascript\n * host.shareContext(oldContext => ({\n * ...oldContext,\n * counter: oldContext.counter + 1\n * }))\n * ```\n */\n shareContext(\n setter: (context: SharedContextValues) => SharedContextValues\n ): void;\n shareContext(\n setter: (context: SharedContextValues) => SharedContextValues\n ): void;\n shareContext(\n setterOrContext:\n | ((context: SharedContextValues) => SharedContextValues)\n | SharedContextValues\n ) {\n if (typeof setterOrContext === \"function\") {\n this.sharedContext = setterOrContext(this.sharedContext);\n } else {\n this.sharedContext = setterOrContext;\n }\n this.emit(\"contextchange\", {\n host: this,\n context: this.sharedContext,\n });\n }\n /**\n * Load extension into host application from provided extension description.\n * Returned promise resolves when all extensions are loaded and registered.\n *\n * @param extensions - List of extension descriptors. Normally, the Host should receive this value from an {@link ExtensionsProvider}.\n * @param options - Custom options to be used as defaults for each {@link Port} object created for each guest.\n * @returns Promise which resolves when all guests have been loaded.\n */\n async load(\n extensions: InstalledExtensions,\n options?: PortOptions\n ): Promise<void> {\n this.runtimeContainer =\n this.runtimeContainer || this.createRuntimeContainer(window);\n const failed: GuestConnection[] = [];\n const loaded: GuestConnection[] = [];\n this.loading = true;\n await Promise.all(\n Object.entries(extensions).map(async ([id, url]) => {\n const port = await this.loadOneGuest(id, url, options);\n (port.error ? failed : loaded).push(port);\n })\n );\n this.loading = false;\n this.emit(\"loadallguests\", { host: this, failed, loaded });\n }\n /**\n * Unload all extensions and remove their frames/workers. Use this to unmount\n * a UI or when switching to a different extensible UI.\n */\n async unload(): Promise<void> {\n this.emit(\"beforeunload\", { host: this });\n await Promise.all([...this.guests.values()].map((guest) => guest.unload()));\n this.guests.clear();\n this.runtimeContainer.parentElement.removeChild(this.runtimeContainer);\n this.emit(\"unload\", { host: this });\n }\n private createRuntimeContainer(window: Window) {\n const { document } = window;\n const container = document.createElement(\"div\");\n container.setAttribute(\"data-uix-guest-container\", this.hostName);\n container.setAttribute(\"role\", \"presentation\");\n container.setAttribute(\"aria-hidden\", \"true\");\n Object.assign(container.style, Host.containerStyle);\n document.body.appendChild(container);\n return container;\n }\n private async loadOneGuest(\n id: string,\n urlString: string,\n options: PortOptions = {}\n ): Promise<GuestConnection> {\n let guest = this.guests.get(id);\n if (!guest) {\n const url = new URL(urlString);\n guest = new Port({\n owner: this.hostName,\n id,\n url,\n runtimeContainer: this.runtimeContainer,\n options: {\n ...this.guestOptions,\n ...options,\n },\n logger: this.logger,\n sharedContext: this.sharedContext,\n events: this as Emits,\n });\n this.guests.set(id, guest);\n }\n this.emit(\"guestbeforeload\", { guest, host: this });\n try {\n await guest.load();\n } catch (e: unknown) {\n const error = new Error(\n `Guest ${guest.id} failed to load: at ${guest.url}: ${\n e instanceof Error ? e.stack : String(e)\n }`\n );\n this.emit(\"error\", { host: this, guest, error });\n return guest;\n }\n // this new guest might have new capabilities, so the identities of the\n // cached capability sets will need to change, to alert subscribers\n this.cachedCapabilityLists = new WeakMap();\n this.emit(\"guestload\", { guest, host: this });\n return guest;\n }\n private getLoadedGuestsWith<Apis extends GuestApis>(\n capabilities: CapabilitySpec<Apis>\n ) {\n if (this.cachedCapabilityLists.has(capabilities)) {\n return this.cachedCapabilityLists.get(capabilities);\n }\n const guestsWithCapabilities = this.getLoadedGuests((guest) =>\n guest.hasCapabilities(capabilities)\n );\n this.cachedCapabilityLists.set(capabilities, guestsWithCapabilities);\n return guestsWithCapabilities;\n }\n}\n","/*\nCopyright 2022 Adobe. All rights reserved.\nThis file is licensed to you under the Apache License, Version 2.0 (the \"License\");\nyou may not use this file except in compliance with the License. You may obtain a copy\nof the License at http://www.apache.org/licenses/LICENSE-2.0\n\nUnless required by applicable law or agreed to in writing, software distributed under\nthe License is distributed on an \"AS IS\" BASIS, WITHOUT WARRANTIES OR REPRESENTATIONS\nOF ANY KIND, either express or implied. See the License for the specific language\ngoverning permissions and limitations under the License.\n*/\n\nimport type {\n Emits,\n GuestConnection,\n HostMethodAddress,\n NamedEvent,\n RemoteHostApis,\n GuestApis,\n CrossRealmObject,\n Unsubscriber,\n VirtualApi,\n} from \"@adobe/uix-core\";\nimport {\n Emitter,\n connectIframe,\n formatHostMethodAddress,\n} from \"@adobe/uix-core\";\nimport { normalizeIframe } from \"./dom-utils\";\n\n/**\n * A specifier for methods to be expected on a remote interface.\n *\n * @remarks\n * A CapabilitySpec is a description of an interface, like a very simplified\n * type definition. It specifies an object structure and the paths in that\n * structure that must be functions. (It doesn't specify anything about the\n * signatures or return values of those functions.)\n *\n * Use CapabilitySpec objects as queries, or filters, to get a subset of\n * installed extensions which have registered methods which match the spec.\n *\n * @example\n * As an extensible app developer, you are making an extension point for spell\n * check. Your code expects extensions to register an API `spellCheck` with\n * methods called `spellCheck.correct(text)` and `spellCheck.suggest(text)`.\n *\n * ```javascript\n * async function correctText(text) {\n * const spellCheckers = host.getLoadedGuests({\n * spellCheck: [\n * 'correct',\n * 'suggest'\n * ]\n * });\n * let correcting = text;\n * for (const checker of spellCheckers) {\n * correcting = await checker.apis.spellCheck.correct(correcting);\n * }\n * return Promise.all(checkers.map(checker =>\n * checker.apis.spellCheck.suggest(correcting)\n * ));\n * }\n * ```\n *\n * @public\n */\nexport type CapabilitySpec<T extends GuestApis> = {\n [Name in keyof T]: (keyof T[Name])[];\n};\n\n/**\n * Interface for decoupling of guest Penpal object\n * @internal\n */\ninterface GuestProxyWrapper {\n // #region Properties (1)\n\n /**\n * Methods from guest\n */\n apis: RemoteHostApis;\n\n // #endregion Properties (1)\n\n // #region Public Methods (1)\n\n /**\n * Emit an event in the guest frame\n */\n emit(type: string, detail: unknown): Promise<void>;\n\n // #endregion Public Methods (1)\n}\n\n/** @public */\ntype PortEvent<\n GuestApi,\n Type extends string = string,\n Detail = Record<string, unknown>\n> = NamedEvent<\n Type,\n Detail &\n Record<string, unknown> & {\n guestPort: Port<GuestApi>;\n }\n>;\n\n/** @public */\nexport type PortEvents<\n GuestApi,\n HostApi extends Record<string, unknown> = Record<string, unknown>\n> =\n | PortEvent<GuestApi, \"hostprovide\">\n | PortEvent<GuestApi, \"unload\">\n | PortEvent<GuestApi, \"beforecallhostmethod\", HostMethodAddress<HostApi>>;\n\n/** @public */\nexport type PortOptions = {\n /**\n * Time in milliseconds to wait for the guest to connect before throwing.\n */\n timeout?: number;\n /**\n * Set true to log copiously in the console.\n */\n debug?: boolean;\n};\n\nconst defaultOptions = {\n timeout: 10000,\n debug: false,\n};\n\n/**\n * A Port is the Host-maintained object representing an extension running as a\n * guest. It exposes methods registered by the Guest, and can provide Host\n * methods back to the guest.\n *\n * @remarks\n * When the Host object loads extensions via {@link Host.load}, it creates a\n * Port object for each extension. When retrieving and filtering extensions\n * via {@link Host.(getLoadedGuests:2)}, a list of Port objects is returned. From\n * the point of view of the extensible app using the Host object, extensions\n * are always Port objects, which expose the methods registered by the\n * extension at the {@link Port.apis} property.\n *\n * @privateRemarks\n * We've gone through several possible names for this object. GuestProxy,\n * GuestInterface, GuestConnection, etc. \"Port\" is not ideal, but it conflicted\n * the least with other types we defined in early drafts. It's definitely\n * something we should review.\n * @public\n */\nexport class Port<GuestApi>\n extends Emitter<PortEvents<GuestApi>>\n implements GuestConnection\n{\n public get apis() {\n if (this.isReady() && this.guestServer) {\n const server = this.guestServer.getRemoteApi();\n return server && server.apis;\n }\n }\n\n // #region Properties (13)\n\n private debug: boolean;\n private logger?: Console;\n private guestServerFrame: HTMLIFrameElement;\n private hostApis: RemoteHostApis = {};\n private isLoaded = false;\n private runtimeContainer: HTMLElement;\n private sharedContext: Record<string, unknown>;\n private subscriptions: Unsubscriber[] = [];\n private timeout: number;\n\n /**\n * Dictionary of namespaced methods that were registered by this guest at the\n * time of connection, using {@link @adobe/uix-guest#register}.\n *\n * @remarks\n * These methods are proxy methods; you can only pass serializable objects to\n * them, not class instances, methods or callbacks.\n * @public\n */\n /**\n * If any errors occurred during the loading of guests, this property will\n * contain the error that was raised.\n * @public\n */\n error?: Error;\n /**\n * The URL of the guest provided by the extension registry. The Host will\n * load this URL in the background, in the invisible the bootstrap frame, so\n * this URL must point to a page that calls {@link @adobe/uix-guest#register}\n * when it loads.\n */\n public url: URL;\n private guestServer: CrossRealmObject<GuestProxyWrapper>;\n\n // #endregion Properties (13)\n\n // #region Constructors (1)\n\n constructor(config: {\n owner: string;\n id: string;\n url: URL;\n /**\n * An alternate DOM element to use for invisible iframes. Will create its\n * own if this option is not populated with a DOM element.\n */\n runtimeContainer: HTMLElement;\n options: PortOptions;\n logger?: Console;\n /**\n * Initial object to populate the shared context with. Once the guest\n * connects, it will be able to access these properties.\n */\n sharedContext: Record<string, unknown>;\n events: Emits;\n }) {\n super(config.id);\n const { timeout, debug } = { ...defaultOptions, ...(config.options || {}) };\n this.timeout = timeout;\n this.debug = debug;\n this.logger = config.logger;\n this.id = config.id;\n this.url = config.url;\n this.runtimeContainer = config.runtimeContainer;\n this.sharedContext = config.sharedContext;\n this.subscriptions.push(\n config.events.addEventListener(\"contextchange\", async (event) => {\n this.sharedContext = (\n (event as CustomEvent).detail as unknown as Record<string, unknown>\n ).context as Record<string, unknown>;\n await this.load();\n await this.guestServer\n .getRemoteApi()\n .emit(\"contextchange\", { context: this.sharedContext });\n })\n );\n }\n\n // #endregion Constructors (1)\n\n // #region Public Methods (6)\n\n /**\n * Connect an iframe element which is displaying another page in the extension\n * with the extension's bootstrap frame, so they can share context and events.\n */\n public attachUI(iframe: HTMLIFrameElement) {\n return this.attachFrame(iframe);\n }\n\n /**\n * Returns true if the guest has registered methods matching the provided\n * capability spec. A capability spec is simply an object whose properties are\n * declared in an array of keys, description the names of the functions and\n * methods that the Port will expose.\n */\n public hasCapabilities(requiredMethods: CapabilitySpec<GuestApis>) {\n this.assertReady();\n return Object.keys(requiredMethods).every((key) => {\n if (!Reflect.has(this.apis, key)) {\n return false;\n }\n const api = this.apis[key];\n const methodList = requiredMethods[\n key as keyof typeof requiredMethods\n ] as string[];\n return methodList.every(\n (methodName: string) =>\n Reflect.has(api, methodName) &&\n typeof api[methodName as keyof typeof api] === \"function\"\n );\n });\n }\n\n /**\n * True when al extensions have loaded.\n */\n public isReady(): boolean {\n return this.isLoaded && !this.error;\n }\n\n /**\n * Loads the extension. Returns a promise which resolves when the extension\n * has loaded. The Host calls this method after retrieving extensions.\n */\n public async load() {\n try {\n if (!this.apis) {\n await this.connect();\n }\n return this.apis;\n } catch (e) {\n this.guestServer = null;\n this.error = e instanceof Error ? e : new Error(String(e));\n throw e;\n }\n }\n\n /**\n * The host-side equivalent of {@link @adobe/uix-guest#register}. Pass a set\n * of methods down to the guest as proxies.\n */\n public provide(apis: RemoteHostApis) {\n Object.assign(this.hostApis, apis);\n this.emit(\"hostprovide\", { guestPort: this, apis });\n }\n\n /**\n * Disconnect from the extension.\n */\n public async unload(): Promise<void> {\n if (this.guestServerFrame && this.guestServerFrame.parentElement) {\n this.guestServerFrame.parentElement.removeChild(this.guestServerFrame);\n this.guestServerFrame = undefined;\n }\n this.emit(\"unload\", { guestPort: this });\n }\n\n // #endregion Public Methods (6)\n\n // #region Private Methods (6)\n\n private assert(\n condition: boolean,\n errorMessage: () => string\n ): asserts condition {\n if (!condition) {\n throw new Error(\n `Error in guest extension \"${this.id}\": ${errorMessage()}`\n );\n }\n }\n\n private assertReady() {\n this.assert(this.isReady(), () => \"Attempted to interact before loaded\");\n }\n\n private attachFrame<T = unknown>(iframe: HTMLIFrameElement) {\n // at least this is necessary\n normalizeIframe(iframe);\n return connectIframe<T>(\n iframe,\n {\n logger: this.logger,\n targetOrigin: this.url.origin,\n timeout: this.timeout,\n },\n {\n getSharedContext: () => this.sharedContext,\n invokeHostMethod: (address: HostMethodAddress) =>\n this.invokeHostMethod(address),\n }\n );\n }\n\n private async connect() {\n const serverFrame =\n this.runtimeContainer.ownerDocument.createElement(\"iframe\");\n normalizeIframe(serverFrame);\n serverFrame.setAttribute(\"aria-hidden\", \"true\");\n serverFrame.setAttribute(\"src\", this.url.href);\n this.guestServerFrame = serverFrame;\n this.runtimeContainer.appendChild(serverFrame);\n if (this.logger) {\n this.logger.info(\n `Guest ${this.id} attached iframe of ${this.url.href}`,\n this\n );\n }\n this.guestServer = await this.attachFrame<GuestProxyWrapper>(serverFrame);\n this.isLoaded = true;\n if (this.logger) {\n this.logger.info(\n `Guest ${this.id} established connection, received methods`,\n this.apis,\n this\n );\n }\n }\n\n private getHostMethodCallee<T = unknown>(\n { name, path }: HostMethodAddress,\n methodSource: RemoteHostApis\n ): RemoteHostApis<VirtualApi> {\n const dots = (level: number) => `host.${path.slice(0, level).join(\".\")}`;\n const methodCallee = path.reduce((current, prop, level) => {\n this.assert(\n Reflect.has(current, prop),\n () => `${dots(level)} has no property \"${prop}\"`\n );\n const next = current[prop];\n this.assert(\n typeof next === \"object\",\n () =>\n `${dots(\n level\n )}.${prop} is not an object; namespaces must be objects with methods`\n );\n return next as RemoteHostApis<GuestApi>;\n }, methodSource);\n this.assert(\n typeof methodCallee[name] === \"function\" &&\n Reflect.has(methodCallee, name),\n () => `\"${dots(path.length - 1)}.${name}\" is not a function`\n );\n return methodCallee;\n }\n\n private invokeHostMethod<T = unknown>(\n address: HostMethodAddress,\n privateMethods?: RemoteHostApis\n ): T {\n const { name, path, args = [] } = address;\n this.assert(name && typeof name === \"string\", () => \"Method name required\");\n this.assert(\n path.length > 0,\n () =>\n `Cannot call a method directly on the host; \".${name}()\" must be in a namespace.`\n );\n let methodCallee;\n if (privateMethods) {\n try {\n methodCallee = this.getHostMethodCallee(address, privateMethods);\n } catch (e) {\n this.logger.warn(\n `Private method ${formatHostMethodAddress(address)} not found!`,\n address\n );\n }\n }\n if (!methodCallee) {\n methodCallee = this.getHostMethodCallee(address, this.hostApis);\n }\n const method = methodCallee[name] as (...args: unknown[]) => T;\n this.emit(\"beforecallhostmethod\", { guestPort: this, name, path, args });\n return method.apply(methodCallee, [\n { id: this.id, url: this.url },\n ...args,\n ]) as T;\n }\n\n // #endregion Private Methods (6)\n}\n","/**\nCopyright 2022 Adobe. All rights reserved.\nThis file is licensed to you under the Apache License, Version 2.0 (the \"License\");\nyou may not use this file except in compliance with the License. You may obtain a copy\nof the License at http://www.apache.org/licenses/LICENSE-2.0\n\nUnless required by applicable law or agreed to in writing, software distributed under\nthe License is distributed on an \"AS IS\" BASIS, WITHOUT WARRANTIES OR REPRESENTATIONS\nOF ANY KIND, either express or implied. See the License for the specific language\ngoverning permissions and limitations under the License.\n*/\n\n/**\n * Strings to be used as values in a space-separated HTML attribute.\n * @internal\n */\nexport type AttrTokens<T> = string | T[];\n\n/**\n * Normalize an argument that may be either a space-separated HTML attribute\n * value or an array of those attribute values into an array of those values.\n * @internal\n */\nexport const tokenizeAttrValues = <T>(tokens: AttrTokens<T>) =>\n (typeof tokens === \"string\" ? tokens.split(\" \") : tokens) as T[];\n\n/**\n * Merge, deduplicate and typecheck a set of strings for an HTML attribute that\n * requires those values to be space-separated tokens.\n * @internal\n *\n * @remarks\n * Useful as a typed interface for any DOM property that is a DOMTokenList.\n * While the DOMTokenList does its own deduplication, it's slightly slower,\n * and while it may do validation of legal tokens, it only does so at runtime.\n * Using {@link AttrTokens} and this function adds typing and autocomplete.\n *\n *\n * @example\n * ```typescript\n * type AllowedClasses =\n * | \"primary\"\n * | \"light\"\n * | \"large\"\n * | \"quiet\";\n *\n * // combine with existing classes and set attribute directly\n * function setStyles(elm: HTMLElement, styles: AttrTokens<AllowedClasses>) {\n * const classNames = mergeAttrValues(\n * elm.className as AttrTokens<AllowedClasses>,\n * styles\n * );\n * elm.className = classNames.join(' ');\n * }\n *\n * // use DOM property directly, but now it's typed!\n * function setSandbox(\n * iframe: HTMLIframeElement,\n * allowedSandboxes: AttrTokens<\"allow-scripts\" | \"allow-popups\">\n * ) {\n * mergeAttrValues(allowedSandboxes).forEach(val => iframe.sandbox.add(val));\n * }\n * ```\n */\nexport const mergeAttrValues = <T>(...tokenLists: AttrTokens<T>[]) => {\n const allMerged = new Set<T>();\n for (const tokenList of tokenLists) {\n for (const token of tokenizeAttrValues(tokenList)) {\n allMerged.add(token);\n }\n }\n return [...allMerged];\n};\n","/**\nCopyright 2022 Adobe. All rights reserved.\nThis file is licensed to you under the Apache License, Version 2.0 (the \"License\");\nyou may not use this file except in compliance with the License. You may obtain a copy\nof the License at http://www.apache.org/licenses/LICENSE-2.0\n\nUnless required by applicable law or agreed to in writing, software distributed under\nthe License is distributed on an \"AS IS\" BASIS, WITHOUT WARRANTIES OR REPRESENTATIONS\nOF ANY KIND, either express or implied. See the License for the specific language\ngoverning permissions and limitations under the License.\n*/\n\nimport { HTMLAttributeReferrerPolicy } from \"react\";\nimport { AttrTokens, mergeAttrValues } from \"./attribute-normalizers\";\n\n/**\n * Sandbox permissions that guest iframes are allowed to have.\n * @internal\n */\ntype SandboxPermission =\n | \"downloads\"\n | \"orientation-lock\"\n | \"pointer-lock\"\n | \"popups\"\n | \"presentation\"\n | \"same-origin\"\n | \"scripts\"\n | \"storage-access-by-user-activation\"\n | \"top-navigation-by-user-activation\";\n\n/**\n * Character strings allowed in \"sandbox=\" attribute.\n * @internal\n */\nexport type SandboxToken = `allow-${SandboxPermission}`;\n\n/**\n * Merge lists of attrs together\n * @internal\n */\nexport const makeSandboxAttrs = (...sandboxes: AttrTokens<SandboxToken>[]) =>\n mergeAttrValues<SandboxToken>(...sandboxes);\n\n/**\n * Limit provided set of \"sandbox\" attributes to a list of legal ones.\n * @internal\n */\nexport const requiredIframeProps = {\n // must not require this until app builder supports CSP\n // csp: \"frame-ancestors 'self'\",\n \"data-uix-guest\": \"true\",\n role: \"presentation\",\n referrerPolicy: \"strict-origin\" as HTMLAttributeReferrerPolicy,\n};\n\nconst requiredIframeAttrEntries = Object.entries(requiredIframeProps);\n\n/**\n * Set and/or verify the required properties for use with uix-sdk on an iframe.\n * @internal\n */\nexport const normalizeIframe = (iframe: HTMLIFrameElement) => {\n for (const [attr, value] of requiredIframeAttrEntries) {\n iframe.setAttribute(attr, value);\n }\n};\n","/*\nCopyright 2022 Adobe. All rights reserved.\nThis file is licensed to you under the Apache License, Version 2.0 (the \"License\");\nyou may not use this file except in compliance with the License. You may obtain a copy\nof the License at http://www.apache.org/licenses/LICENSE-2.0\n\nUnless required by applicable law or agreed to in writing, software distributed under\nthe License is distributed on an \"AS IS\" BASIS, WITHOUT WARRANTIES OR REPRESENTATIONS\nOF ANY KIND, either express or implied. See the License for the specific language\ngoverning permissions and limitations under the License.\n*/\n\n/* eslint-disable @typescript-eslint/restrict-template-expressions */\n/* eslint-disable @typescript-eslint/no-unsafe-member-access */\n/**\n * Adapter to attach console logging listeners to a Host running in an app\n * @hidden\n */\nimport {\n debugEmitter,\n EmitterDebugLogger,\n Emits,\n GuestConnection,\n} from \"@adobe/uix-core\";\nimport type { PortEvents } from \"./port.js\";\nimport type { HostEventLoadAllGuests, HostEvents } from \"./host.js\";\n\ntype GenericPortEvents = PortEvents<Record<string, unknown>>;\n\ntype Portlike = GuestConnection & Emits<GenericPortEvents>;\n\nexport function debugHost(host: Emits<HostEvents>): EmitterDebugLogger {\n const hostLogger = debugEmitter(host, {\n theme: \"blue medium\",\n type: \"Host\",\n });\n hostLogger\n .listen(\"guestbeforeload\", (log, event) => {\n const { detail } = event;\n const guest = detail.guest as Portlike;\n log.info(event, `Guest ID ${guest.id}`);\n const portLogger = debugEmitter(guest, {\n theme: \"green medium\",\n type: \"Port\",\n id: `${host.id} ➔ ${guest.id}`,\n });\n portLogger\n .listen(\"hostprovide\", (log, event) => {\n log.info(\"received APIs\", event.detail.apis);\n })\n .listen(\"beforecallhostmethod\", (log, event) => {\n log.info(event.detail);\n })\n .listen(\"unload\", (log, event) => {\n log.info(event.detail);\n log.detach();\n });\n })\n .listen(\"guestload\", (log, e) => {\n log.info(e.detail.guest.id, e.detail.guest);\n })\n .listen(\"error\", (log, e) => {\n log.error(`Error: ${e.detail.error.message}`, e);\n })\n .listen(\n \"loadallguests\",\n (log, { detail: { failed, loaded, host } }: HostEventLoadAllGuests) => {\n if (failed.length > 0) {\n log.error(\"%d guests failed to load!\", failed.length);\n }\n log.info(\"%d guests loaded\", loaded.length, host);\n }\n )\n .listen(\"unload\", (log) => {\n log.info(\"Unloaded guest and container.\");\n log.detach();\n });\n return hostLogger;\n}\n","/*\nCopyright 2022 Adobe. All rights reserved.\nThis file is licensed to you under the Apache License, Version 2.0 (the \"License\");\nyou may not use this file except in compliance with the License. You may obtain a copy\nof the License at http://www.apache.org/licenses/LICENSE-2.0\n\nUnless required by applicable law or agreed to in writing, software distributed under\nthe License is distributed on an \"AS IS\" BASIS, WITHOUT WARRANTIES OR REPRESENTATIONS\nOF ANY KIND, either express or implied. See the License for the specific language\ngoverning permissions and limitations under the License.\n*/\n\nimport { InstalledExtensions, ExtensionsProvider } from \"../host.js\";\n\ninterface ExtensionDefinition {\n name: string;\n title: string;\n description: string;\n icon: string;\n publisher: string;\n endpoints: Record<string, EndpointDefinition>;\n xrInfo: ExtensionInfo;\n status: string;\n}\n\ntype EndpointDefinition = Record<string, Array<OperationDefinition>>;\n\ninterface ExtensionInfo {\n supportEmail: string;\n appId: string;\n}\n\ninterface OperationDefinition {\n href: string;\n metadata: OperationMetadata;\n}\n\ninterface OperationMetadata {\n services: Array<object>;\n profile: OperationProfile;\n}\n\ninterface OperationProfile {\n client_id: string;\n scope: string;\n}\n\n/** @public */\nexport interface ExtensionRegistryEndpointRegistration {\n service: string;\n extensionPoint: string;\n version: string;\n}\n\n/** @public */\nexport interface ExtensionRegistryExtensionRegistration\n extends ExtensionRegistryEndpointRegistration {\n imsOrg: string;\n}\n\n/** @public */\nexport interface ExtensionRegistryConnection {\n baseUrl?: string;\n apiKey: string;\n auth: {\n schema: \"Basic\" | \"Bearer\";\n imsToken: string;\n };\n}\n\n/** @public */\nexport interface ExtensionRegistryConfig\n extends ExtensionRegistryExtensionRegistration,\n ExtensionRegistryConnection {}\n\nfunction buildEndpointPath(\n config: ExtensionRegistryEndpointRegistration\n): string {\n return `${config.service}/${config.extensionPoint}/${config.version}`;\n}\n\nfunction ensureProtocolSpecified(url: string) {\n if (url.startsWith(\"https://\")) {\n return url;\n }\n if (url.startsWith(\"http://\")) {\n return url;\n }\n return `https://${url}`;\n}\n\nasync function fetchExtensionsFromRegistry(\n config: ExtensionRegistryConfig\n): Promise<Array<ExtensionDefinition>> {\n const resp = await fetch(\n `${ensureProtocolSpecified(\n config.baseUrl || \"appregistry.adobe.io\"\n )}/myxchng/v1/org/${encodeURIComponent(\n config.imsOrg\n )}/xtn/${buildEndpointPath(config)}?auth=true`,\n {\n headers: {\n Accept: \"application/json\",\n Authorization: `${config.auth.schema} ${config.auth.imsToken}`, // todo: check if auth schema needed (initial implementation was without it)\n \"X-Api-Key\": config.apiKey,\n },\n }\n );\n\n if (resp.status != 200) {\n throw new Error(\n `extension registry returned non-200 response (${\n resp.status\n }): ${await resp.text()}`\n );\n }\n\n return await resp.json();\n}\n\nfunction extensionRegistryExtensionsProvider(\n config: ExtensionRegistryConfig\n): Promise<InstalledExtensions> {\n const erEndpoint = buildEndpointPath(config);\n return fetchExtensionsFromRegistry(config).then((out) =>\n out.reduce((a, e: ExtensionDefinition) => {\n if (e.status !== \"PUBLISHED\") {\n return a;\n }\n\n return {\n ...a,\n // todo: make safer way to extract href\n [e.name]: e.endpoints[erEndpoint].view[0].href,\n };\n }, {})\n );\n\n return Promise.resolve({});\n}\n\n/**\n * Create a callback that fetches extensions from the registry.\n * @public\n */\nexport function createExtensionRegistryProvider(\n config: ExtensionRegistryConfig\n): ExtensionsProvider {\n return function () {\n return extensionRegistryExtensionsProvider(config);\n };\n}\n","/*\nCopyright 2022 Adobe. All rights reserved.\nThis file is licensed to you under the Apache License, Version 2.0 (the \"License\");\nyou may not use this file except in compliance with the License. You may obtain a copy\nof the License at http://www.apache.org/licenses/LICENSE-2.0\n\nUnless required by applicable law or agreed to in writing, software distributed under\nthe License is distributed on an \"AS IS\" BASIS, WITHOUT WARRANTIES OR REPRESENTATIONS\nOF ANY KIND, either express or implied. See the License for the specific language\ngoverning permissions and limitations under the License.\n*/\n\nimport { InstalledExtensions, ExtensionsProvider } from \"../host.js\";\n\n/**\n * Combine multiple {@link @adobe/uix-host#ExtensionsProvider} callbacks into a\n * single callback, which aggregates and dedupes all extensions from all\n * providers into one namespaced object.\n * @public\n */\nexport function combineExtensionsFromProviders(\n ...providers: Array<ExtensionsProvider>\n): ExtensionsProvider {\n return () =>\n Promise.all(providers.map((ep: ExtensionsProvider) => ep())).then(\n (extensionsBatches: Array<InstalledExtensions>) => {\n return Object.assign({}, ...extensionsBatches);\n }\n );\n}\n","/*\nCopyright 2022 Adobe. All rights reserved.\nThis file is licensed to you under the Apache License, Version 2.0 (the \"License\");\nyou may not use this file except in compliance with the License. You may obtain a copy\nof the License at http://www.apache.org/licenses/LICENSE-2.0\n\nUnless required by applicable law or agreed to in writing, software distributed under\nthe License is distributed on an \"AS IS\" BASIS, WITHOUT WARRANTIES OR REPRESENTATIONS\nOF ANY KIND, either express or implied. See the License for the specific language\ngoverning permissions and limitations under the License.\n*/\n\nimport { ExtensionsProvider } from \"../host.js\";\n\n/**\n * Mute any errors produced by provider.\n * This function would execute given provider and return its results as is, if any error occurs this provider will log it\n * any return an empty array of extensions.\n * @public\n */\nexport function mutedProvider(\n provider: ExtensionsProvider\n): ExtensionsProvider {\n return async () => {\n try {\n return await provider();\n } catch (error) {\n console.error(`Extension provider has failed: ${error.message}`, {\n error,\n });\n return {};\n }\n };\n}\n"]}
|
1
|
+
{"version":3,"sources":["../src/host.ts","../src/port.ts","../src/dom-utils/attribute-normalizers.ts","../src/dom-utils/iframe-normalizers.ts","../src/debug-host.ts","../src/extensions-provider/extension-registry.ts","../src/extensions-provider/composition.ts","../src/extensions-provider/mute.ts"],"names":["Emitter","log","event","host","window"],"mappings":";AAoBA,SAAS,WAAAA,UAAS,oBAAoB;;;ACItC;AAAA,EACE;AAAA,EACA;AAAA,EACA;AAAA,OACK;;;ACLA,IAAM,qBAAqB,CAAI,WACnC,OAAO,WAAW,WAAW,OAAO,MAAM,GAAG,IAAI;AAwC7C,IAAM,kBAAkB,IAAO,eAAgC;AACpE,QAAM,YAAY,oBAAI,IAAO;AAC7B,aAAW,aAAa,YAAY;AAClC,eAAW,SAAS,mBAAmB,SAAS,GAAG;AACjD,gBAAU,IAAI,KAAK;AAAA,IACrB;AAAA,EACF;AACA,SAAO,CAAC,GAAG,SAAS;AACtB;;;AChCO,IAAM,mBAAmB,IAAI,cAClC,gBAA8B,GAAG,SAAS;AAMrC,IAAM,sBAAsB;AAAA;AAAA;AAAA,EAGjC,kBAAkB;AAAA,EAClB,MAAM;AAAA,EACN,gBAAgB;AAClB;AAEA,IAAM,4BAA4B,OAAO,QAAQ,mBAAmB;AAM7D,IAAM,kBAAkB,CAAC,WAA8B;AAC5D,aAAW,CAAC,MAAM,KAAK,KAAK,2BAA2B;AACrD,WAAO,aAAa,MAAM,KAAK;AAAA,EACjC;AACF;;;AF2CA,IAAM,iBAAiB;AAAA,EACrB,SAAS;AAAA,EACT,OAAO;AACT;AAsBO,IAAM,OAAN,cACG,QAEV;AAAA;AAAA;AAAA,EAgDE,YAAY,QAiBT;AACD,UAAM,OAAO,EAAE;AArDjB,SAAQ,WAA2B,CAAC;AACpC,SAAQ,WAAW;AAGnB,SAAQ,gBAAgC,CAAC;AAkDvC,UAAM,EAAE,SAAS,MAAM,IAAI,EAAE,GAAG,gBAAgB,GAAI,OAAO,WAAW,CAAC,EAAG;AAC1E,SAAK,UAAU;AACf,SAAK,QAAQ;AACb,SAAK,SAAS,OAAO;AACrB,SAAK,KAAK,OAAO;AACjB,SAAK,MAAM,OAAO;AAClB,SAAK,mBAAmB,OAAO;AAC/B,SAAK,gBAAgB,OAAO;AAC5B,SAAK,cAAc;AAAA,MACjB,OAAO,OAAO,iBAAiB,iBAAiB,OAAO,UAAU;AAC/D,aAAK,gBACF,MAAsB,OACvB;AACF,cAAM,KAAK,KAAK;AAChB,cAAM,KAAK,YACR,aAAa,EACb,KAAK,iBAAiB,EAAE,SAAS,KAAK,cAAc,CAAC;AAAA,MAC1D,CAAC;AAAA,IACH;AAAA,EACF;AAAA,EArFA,IAAW,OAAO;AAChB,QAAI,KAAK,QAAQ,KAAK,KAAK,aAAa;AACtC,YAAM,SAAS,KAAK,YAAY,aAAa;AAC7C,aAAO,UAAU,OAAO;AAAA,IAC1B;AAAA,EACF;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EA0FO,SACL,QAC8B;AAC9B,WAAO,KAAK,YAAY,QAAQ;AAAA,MAC9B,gBAAgB,CAAC,eAAkD;AACjE,aAAK,KAAK,eAAe;AAAA,UACvB;AAAA,UACA,WAAW;AAAA,UACX;AAAA,QACF,CAAC;AAAA,MACH;AAAA,IACF,CAAkB;AAAA,EACpB;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAQO,gBAAgB,iBAA4C;AACjE,SAAK,YAAY;AACjB,WAAO,OAAO,KAAK,eAAe,EAAE,MAAM,CAAC,QAAQ;AACjD,UAAI,CAAC,QAAQ,IAAI,KAAK,MAAM,GAAG,GAAG;AAChC,eAAO;AAAA,MACT;AACA,YAAM,MAAM,KAAK,KAAK,GAAG;AACzB,YAAM,aAAa,gBACjB,GACF;AACA,aAAO,WAAW;AAAA,QAChB,CAAC,eACC,QAAQ,IAAI,KAAK,UAAU,KAC3B,OAAO,IAAI,UAA8B,MAAM;AAAA,MACnD;AAAA,IACF,CAAC;AAAA,EACH;AAAA;AAAA;AAAA;AAAA,EAKO,UAAmB;AACxB,WAAO,KAAK,YAAY,CAAC,KAAK;AAAA,EAChC;AAAA;AAAA;AAAA;AAAA;AAAA,EAMA,MAAa,OAAO;AAClB,QAAI;AACF,UAAI,CAAC,KAAK,MAAM;AACd,cAAM,KAAK,QAAQ;AAAA,MACrB;AACA,aAAO,KAAK;AAAA,IACd,SAAS,GAAP;AACA,WAAK,cAAc;AACnB,WAAK,QAAQ,aAAa,QAAQ,IAAI,IAAI,MAAM,OAAO,CAAC,CAAC;AACzD,YAAM;AAAA,IACR;AAAA,EACF;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOO,QAAQ,MAAsB;AACnC,eAAW,CAAC,cAAc,OAAO,KAAK,OAAO,QAAQ,IAAI,GAAG;AAC1D,WAAK,SAAS,YAAY,IAAI,KAAK,SAAS,YAAY,KAAK,CAAC;AAC9D,aAAO,OAAO,KAAK,SAAS,YAAY,GAAG,OAAO;AAAA,IACpD;AACA,SAAK,KAAK,eAAe,EAAE,WAAW,MAAM,KAAK,CAAC;AAAA,EACpD;AAAA;AAAA;AAAA;AAAA,EAKA,MAAa,SAAwB;AACnC,QAAI,KAAK,oBAAoB,KAAK,iBAAiB,eAAe;AAChE,WAAK,iBAAiB,cAAc,YAAY,KAAK,gBAAgB;AACrE,WAAK,mBAAmB;AAAA,IAC1B;AACA,SAAK,KAAK,UAAU,EAAE,WAAW,KAAK,CAAC;AAAA,EACzC;AAAA;AAAA;AAAA,EAMQ,OACN,WACA,cACmB;AACnB,QAAI,CAAC,WAAW;AACd,YAAM,IAAI;AAAA,QACR,6BAA6B,KAAK,QAAQ,aAAa;AAAA,MACzD;AAAA,IACF;AAAA,EACF;AAAA,EAEQ,cAAc;AACpB,SAAK,OAAO,KAAK,QAAQ,GAAG,MAAM,qCAAqC;AAAA,EACzE;AAAA,EAEQ,YACN,QACA,eAAuB,CAAC,GACxB;AAEA,oBAAgB,MAAM;AACtB,SAAK,OAAO,IAAI,eAAe,MAAM;AACrC,WAAO;AAAA,MACL;AAAA,MACA;AAAA,QACE,QAAQ,KAAK;AAAA,QACb,cAAc,KAAK,IAAI;AAAA,QACvB,SAAS,KAAK;AAAA,MAChB;AAAA,MACA;AAAA,QACE,kBAAkB,MAAM,KAAK;AAAA,QAC7B,kBAAkB,CAAC,YACjB,KAAK,iBAAiB,OAAO;AAAA,QAC/B,GAAG;AAAA,MACL;AAAA,IACF;AAAA,EACF;AAAA,EAEA,MAAc,UAAU;AACtB,UAAM,cACJ,KAAK,iBAAiB,cAAc,cAAc,QAAQ;AAC5D,oBAAgB,WAAW;AAC3B,gBAAY,aAAa,eAAe,MAAM;AAC9C,gBAAY,aAAa,OAAO,KAAK,IAAI,IAAI;AAC7C,SAAK,mBAAmB;AACxB,SAAK,iBAAiB,YAAY,WAAW;AAC7C,QAAI,KAAK,QAAQ;AACf,WAAK,OAAO;AAAA,QACV,SAAS,KAAK,yBAAyB,KAAK,IAAI;AAAA,QAChD;AAAA,MACF;AAAA,IACF;AACA,SAAK,cAAc,MAAM,KAAK,YAA+B,WAAW;AACxE,SAAK,WAAW;AAChB,QAAI,KAAK,QAAQ;AACf,WAAK,OAAO;AAAA,QACV,SAAS,KAAK;AAAA,QACd,KAAK;AAAA,QACL;AAAA,MACF;AAAA,IACF;AAAA,EACF;AAAA,EAEQ,oBACN,EAAE,MAAM,KAAK,GACb,cAC4B;AAC5B,UAAM,OAAO,CAAC,UAAkB,QAAQ,KAAK,MAAM,GAAG,KAAK,EAAE,KAAK,GAAG;AACrE,UAAM,eAAe,KAAK,OAAO,CAAC,SAAS,MAAM,UAAU;AACzD,WAAK;AAAA,QACH,QAAQ,IAAI,SAAS,IAAI;AAAA,QACzB,MAAM,GAAG,KAAK,KAAK,sBAAsB;AAAA,MAC3C;AACA,YAAM,OAAO,QAAQ,IAAI;AACzB,WAAK;AAAA,QACH,OAAO,SAAS;AAAA,QAChB,MACE,GAAG;AAAA,UACD;AAAA,QACF,KAAK;AAAA,MACT;AACA,aAAO;AAAA,IACT,GAAG,YAAY;AACf,SAAK;AAAA,MACH,OAAO,aAAa,IAAI,MAAM,cAC5B,QAAQ,IAAI,cAAc,IAAI;AAAA,MAChC,MAAM,IAAI,KAAK,KAAK,SAAS,CAAC,KAAK;AAAA,IACrC;AACA,WAAO;AAAA,EACT;AAAA,EAEQ,iBACN,SACA,gBACG;AACH,UAAM,EAAE,MAAM,MAAM,OAAO,CAAC,EAAE,IAAI;AAClC,SAAK,OAAO,QAAQ,OAAO,SAAS,UAAU,MAAM,sBAAsB;AAC1E,SAAK;AAAA,MACH,KAAK,SAAS;AAAA,MACd,MACE,gDAAgD;AAAA,IACpD;AACA,QAAI;AACJ,QAAI,gBAAgB;AAClB,UAAI;AACF,uBAAe,KAAK,oBAAoB,SAAS,cAAc;AAAA,MACjE,SAAS,GAAP;AACA,aAAK,OAAO;AAAA,UACV,kBAAkB,wBAAwB,OAAO;AAAA,UACjD;AAAA,QACF;AAAA,MACF;AAAA,IACF;AACA,QAAI,CAAC,cAAc;AACjB,qBAAe,KAAK,oBAAoB,SAAS,KAAK,QAAQ;AAAA,IAChE;AACA,UAAM,SAAS,aAAa,IAAI;AAChC,SAAK,KAAK,wBAAwB,EAAE,WAAW,MAAM,MAAM,MAAM,KAAK,CAAC;AACvE,WAAO,OAAO,MAAM,cAAc;AAAA,MAChC,EAAE,IAAI,KAAK,IAAI,KAAK,KAAK,IAAI;AAAA,MAC7B,GAAG;AAAA,IACL,CAAC;AAAA,EACH;AAAA;AAGF;;;AG7aA;AAAA,EACE;AAAA,OAIK;AAGA,SAAS,UAAU,MAA6C;AACrE,QAAM,aAAa,aAAa,MAAM;AAAA,IACpC,OAAO;AAAA,IACP,MAAM;AAAA,EACR,CAAC;AACD,aACG,OAAO,mBAAmB,CAAC,KAAK,UAAU;AACzC,UAAM,EAAE,OAAO,IAAI;AACnB,UAAM,QAAQ,OAAO;AACrB,QAAI,KAAK,OAAO,YAAY,MAAM,IAAI;AACtC,UAAM,aAAa,aAAa,OAAO;AAAA,MACrC,OAAO;AAAA,MACP,MAAM;AAAA,MACN,IAAI,GAAG,KAAK,aAAQ,MAAM;AAAA,IAC5B,CAAC;AACD,eACG,OAAO,eAAe,CAACC,MAAKC,WAAU;AACrC,MAAAD,KAAI,KAAK,iBAAiBC,OAAM,OAAO,IAAI;AAAA,IAC7C,CAAC,EACA,OAAO,wBAAwB,CAACD,MAAKC,WAAU;AAC9C,MAAAD,KAAI,KAAKC,OAAM,MAAM;AAAA,IACvB,CAAC,EACA,OAAO,eAAe,CAACD,MAAKC,WAAU;AACrC,MAAAD,KAAI,KAAKC,OAAM,MAAM;AAAA,IACvB,CAAC,EACA,OAAO,UAAU,CAACD,MAAKC,WAAU;AAChC,MAAAD,KAAI,KAAKC,OAAM,MAAM;AACrB,MAAAD,KAAI,OAAO;AAAA,IACb,CAAC;AAAA,EACL,CAAC,EACA,OAAO,aAAa,CAAC,KAAK,MAAM;AAC/B,QAAI,KAAK,EAAE,OAAO,MAAM,IAAI,EAAE,OAAO,KAAK;AAAA,EAC5C,CAAC,EACA,OAAO,SAAS,CAAC,KAAK,MAAM;AAC3B,QAAI,MAAM,UAAU,EAAE,OAAO,MAAM,WAAW,CAAC;AAAA,EACjD,CAAC,EACA;AAAA,IACC;AAAA,IACA,CAAC,KAAK,EAAE,QAAQ,EAAE,QAAQ,QAAQ,MAAAE,MAAK,EAAE,MAA8B;AACrE,UAAI,OAAO,SAAS,GAAG;AACrB,YAAI,MAAM,6BAA6B,OAAO,MAAM;AAAA,MACtD;AACA,UAAI,KAAK,oBAAoB,OAAO,QAAQA,KAAI;AAAA,IAClD;AAAA,EACF,EACC,OAAO,UAAU,CAAC,QAAQ;AACzB,QAAI,KAAK,+BAA+B;AACxC,QAAI,OAAO;AAAA,EACb,CAAC;AACH,SAAO;AACT;;;AJiDA,IAAM,gBAAgB,MAAM;AA0BrB,IAAM,QAAN,cAAmBH,SAAoB;AAAA,EAwE5C,YAAY,QAAoB;AAC9B,UAAM,OAAO,QAAQ;AAXvB;AAAA;AAAA;AAAA;AAAA,mBAAU;AAIV;AAAA;AAAA;AAAA,kBAAkB,oBAAI,IAAI;AAC1B,SAAQ,wBAAiD,oBAAI,QAAQ;AAGrE,SAAQ,SAAkB;AAIxB,UAAM,EAAE,eAAe,CAAC,EAAE,IAAI;AAC9B,SAAK,eAAe;AAAA,MAClB,GAAG;AAAA,MACH,OAAO,aAAa,UAAU,QAAQ,QAAQ,CAAC,CAAC,OAAO;AAAA,IACzD;AACA,SAAK,WAAW,OAAO;AACvB,SAAK,gBAAgB,OAAO,iBAAiB,CAAC;AAC9C,SAAK,mBAAmB,OAAO;AAC/B,QAAI,OAAO,OAAO;AAChB,WAAK,SAAS,UAAU,IAAI;AAAA,IAC9B;AAAA,EACF;AAAA,EAeA,gBACE,sBACmB;AACnB,QAAI,OAAO,yBAAyB,UAAU;AAC5C,aAAO,KAAK,oBAA0B,oBAAoB;AAAA,IAC5D;AACA,UAAM,SAAS,wBAAwB;AACvC,UAAM,SAAS,CAAC;AAChB,eAAW,SAAS,KAAK,OAAO,OAAO,GAAG;AACxC,UAAI,MAAM,QAAQ,KAAK,OAAO,KAAK,GAAG;AACpC,eAAO,KAAK,KAAwB;AAAA,MACtC;AAAA,IACF;AACA,WAAO;AAAA,EACT;AAAA,EAoDA,aACE,iBAGA;AACA,QAAI,OAAO,oBAAoB,YAAY;AACzC,WAAK,gBAAgB,gBAAgB,KAAK,aAAa;AAAA,IACzD,OAAO;AACL,WAAK,gBAAgB;AAAA,IACvB;AACA,SAAK,KAAK,iBAAiB;AAAA,MACzB,MAAM;AAAA,MACN,SAAS,KAAK;AAAA,IAChB,CAAC;AAAA,EACH;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EASA,MAAM,KACJ,YACA,SACe;AACf,SAAK,mBACH,KAAK,oBAAoB,KAAK,uBAAuB,MAAM;AAC7D,UAAM,SAAiB,CAAC;AACxB,UAAM,SAAiB,CAAC;AACxB,SAAK,UAAU;AACf,UAAM,QAAQ;AAAA,MACZ,OAAO,QAAQ,UAAU,EAAE,IAAI,OAAO,CAAC,IAAI,GAAG,MAAM;AAClD,cAAM,OAAO,MAAM,KAAK,aAAa,IAAI,KAAK,OAAO;AACrD,SAAC,KAAK,QAAQ,SAAS,QAAQ,KAAK,IAAI;AAAA,MAC1C,CAAC;AAAA,IACH;AACA,SAAK,UAAU;AACf,SAAK,KAAK,iBAAiB,EAAE,MAAM,MAAM,QAAQ,OAAO,CAAC;AAAA,EAC3D;AAAA;AAAA;AAAA;AAAA;AAAA,EAKA,MAAM,SAAwB;AAC5B,SAAK,KAAK,gBAAgB,EAAE,MAAM,KAAK,CAAC;AACxC,UAAM,QAAQ,IAAI,CAAC,GAAG,KAAK,OAAO,OAAO,CAAC,EAAE,IAAI,CAAC,UAAU,MAAM,OAAO,CAAC,CAAC;AAC1E,SAAK,OAAO,MAAM;AAClB,SAAK,iBAAiB,cAAc,YAAY,KAAK,gBAAgB;AACrE,SAAK,KAAK,UAAU,EAAE,MAAM,KAAK,CAAC;AAAA,EACpC;AAAA,EACQ,uBAAuBI,SAAgB;AAC7C,UAAM,EAAE,SAAS,IAAIA;AACrB,UAAM,YAAY,SAAS,cAAc,KAAK;AAC9C,cAAU,aAAa,4BAA4B,KAAK,QAAQ;AAChE,cAAU,aAAa,QAAQ,cAAc;AAC7C,cAAU,aAAa,eAAe,MAAM;AAC5C,WAAO,OAAO,UAAU,OAAO,MAAK,cAAc;AAClD,aAAS,KAAK,YAAY,SAAS;AACnC,WAAO;AAAA,EACT;AAAA,EACA,MAAc,aACZ,IACA,WACA,UAAuB,CAAC,GACN;AAClB,QAAI,QAAQ,KAAK,OAAO,IAAI,EAAE;AAC9B,QAAI,CAAC,OAAO;AACV,YAAM,MAAM,IAAI,IAAI,SAAS;AAC7B,cAAQ,IAAI,KAAK;AAAA,QACf,OAAO,KAAK;AAAA,QACZ;AAAA,QACA;AAAA,QACA,kBAAkB,KAAK;AAAA,QACvB,SAAS;AAAA,UACP,GAAG,KAAK;AAAA,UACR,GAAG;AAAA,QACL;AAAA,QACA,QAAQ,KAAK;AAAA,QACb,eAAe,KAAK;AAAA,QACpB,QAAQ;AAAA,MACV,CAAC;AACD,WAAK,OAAO,IAAI,IAAI,KAAK;AAAA,IAC3B;AACA,SAAK,KAAK,mBAAmB,EAAE,OAAO,MAAM,KAAK,CAAC;AAClD,QAAI;AACF,YAAM,MAAM,KAAK;AAAA,IACnB,SAAS,GAAP;AACA,YAAM,QAAQ,IAAI;AAAA,QAChB,SAAS,MAAM,yBAAyB,MAAM,QAC5C,aAAa,QAAQ,EAAE,QAAQ,OAAO,CAAC;AAAA,MAE3C;AACA,WAAK,KAAK,SAAS,EAAE,MAAM,MAAM,OAAO,MAAM,CAAC;AAC/C,aAAO;AAAA,IACT;AAGA,SAAK,wBAAwB,oBAAI,QAAQ;AACzC,SAAK,KAAK,aAAa,EAAE,OAAO,MAAM,KAAK,CAAC;AAC5C,WAAO;AAAA,EACT;AAAA,EACQ,oBACN,cACA;AACA,QAAI,KAAK,sBAAsB,IAAI,YAAY,GAAG;AAChD,aAAO,KAAK,sBAAsB,IAAI,YAAY;AAAA,IACpD;AACA,UAAM,yBAAyB,KAAK;AAAA,MAAgB,CAAC,UACnD,MAAM,gBAAgB,YAAY;AAAA,IACpC;AACA,SAAK,sBAAsB,IAAI,cAAc,sBAAsB;AACnE,WAAO;AAAA,EACT;AACF;AAzRO,IAAM,OAAN;AAAM,KA6CI,iBAAiB;AAAA,EAC9B,UAAU;AAAA,EACV,OAAO;AAAA,EACP,QAAQ;AAAA,EACR,eAAe;AAAA,EACf,SAAS;AAAA,EACT,KAAK;AAAA,EACL,MAAM;AACR;;;AKjIF,SAAS,kBACP,QACQ;AACR,SAAO,GAAG,OAAO,WAAW,OAAO,kBAAkB,OAAO;AAC9D;AAEA,SAAS,wBAAwB,KAAa;AAC5C,MAAI,IAAI,WAAW,UAAU,GAAG;AAC9B,WAAO;AAAA,EACT;AACA,MAAI,IAAI,WAAW,SAAS,GAAG;AAC7B,WAAO;AAAA,EACT;AACA,SAAO,WAAW;AACpB;AAEA,eAAe,4BACb,QACqC;AACrC,QAAM,OAAO,MAAM;AAAA,IACjB,GAAG;AAAA,MACD,OAAO,WAAW;AAAA,IACpB,oBAAoB;AAAA,MAClB,OAAO;AAAA,IACT,SAAS,kBAAkB,MAAM;AAAA,IACjC;AAAA,MACE,SAAS;AAAA,QACP,QAAQ;AAAA,QACR,eAAe,GAAG,OAAO,KAAK,UAAU,OAAO,KAAK;AAAA;AAAA,QACpD,aAAa,OAAO;AAAA,MACtB;AAAA,IACF;AAAA,EACF;AAEA,MAAI,KAAK,UAAU,KAAK;AACtB,UAAM,IAAI;AAAA,MACR,iDACE,KAAK,YACD,MAAM,KAAK,KAAK;AAAA,IACxB;AAAA,EACF;AAEA,SAAO,MAAM,KAAK,KAAK;AACzB;AAEA,SAAS,oCACP,QAC8B;AAC9B,QAAM,aAAa,kBAAkB,MAAM;AAC3C,SAAO,4BAA4B,MAAM,EAAE;AAAA,IAAK,CAAC,QAC/C,IAAI,OAAO,CAAC,GAAG,MAA2B;AACxC,UAAI,EAAE,WAAW,aAAa;AAC5B,eAAO;AAAA,MACT;AAEA,aAAO;AAAA,QACL,GAAG;AAAA;AAAA,QAEH,CAAC,EAAE,IAAI,GAAG,EAAE,UAAU,UAAU,EAAE,KAAK,CAAC,EAAE;AAAA,MAC5C;AAAA,IACF,GAAG,CAAC,CAAC;AAAA,EACP;AAEA,SAAO,QAAQ,QAAQ,CAAC,CAAC;AAC3B;AAMO,SAAS,gCACd,QACoB;AACpB,SAAO,WAAY;AACjB,WAAO,oCAAoC,MAAM;AAAA,EACnD;AACF;;;ACnIO,SAAS,kCACX,WACiB;AACpB,SAAO,MACL,QAAQ,IAAI,UAAU,IAAI,CAAC,OAA2B,GAAG,CAAC,CAAC,EAAE;AAAA,IAC3D,CAAC,sBAAkD;AACjD,aAAO,OAAO,OAAO,CAAC,GAAG,GAAG,iBAAiB;AAAA,IAC/C;AAAA,EACF;AACJ;;;ACTO,SAAS,cACd,UACoB;AACpB,SAAO,YAAY;AACjB,QAAI;AACF,aAAO,MAAM,SAAS;AAAA,IACxB,SAAS,OAAP;AACA,cAAQ,MAAM,kCAAkC,MAAM,WAAW;AAAA,QAC/D;AAAA,MACF,CAAC;AACD,aAAO,CAAC;AAAA,IACV;AAAA,EACF;AACF","sourcesContent":["/*\nCopyright 2022 Adobe. All rights reserved.\nThis file is licensed to you under the Apache License, Version 2.0 (the \"License\");\nyou may not use this file except in compliance with the License. You may obtain a copy\nof the License at http://www.apache.org/licenses/LICENSE-2.0\n\nUnless required by applicable law or agreed to in writing, software distributed under\nthe License is distributed on an \"AS IS\" BASIS, WITHOUT WARRANTIES OR REPRESENTATIONS\nOF ANY KIND, either express or implied. See the License for the specific language\ngoverning permissions and limitations under the License.\n*/\n\nimport type {\n Extension,\n GuestEmitter,\n NamedEvent,\n Emits,\n GuestApis,\n} from \"@adobe/uix-core\";\nimport type { CapabilitySpec } from \"./port.js\";\nimport { Emitter, quietConsole } from \"@adobe/uix-core\";\nimport { Port, PortOptions } from \"./port.js\";\nimport { debugHost } from \"./debug-host.js\";\n\n/**\n * Dictionary of {@link Port} objects by extension ID.\n * @public\n */\nexport type PortMap = Map<string, Port>;\n\n/** @public */\nexport type HostEvent<\n Type extends string = string,\n Detail = Record<string, unknown>\n> = NamedEvent<Type, Detail & Record<string, unknown> & { host: Host }>;\n/** @public */\ntype HostGuestEvent<Type extends string> = HostEvent<\n `guest${Type}`,\n { guest: Port }\n>;\n\n/**\n * All guests requested by host have been loaded and connected.\n * @public\n */\nexport type HostEventLoadAllGuests = HostEvent<\n \"loadallguests\",\n { failed: Port[]; loaded: Port[] }\n>;\n\n/**\n * Shared context has been set or updated; all guests receive this event too.\n * @public\n */\n\nexport type HostEventContextChange = HostEvent<\n \"contextchange\",\n { context: SharedContextValues }\n>;\n\n/**\n * An error has occurred during loading or unloading of guests.\n * @public\n */\nexport type HostEventError = HostEvent<\"error\", { error: Error }>;\n\n/** @public */\nexport type HostEvents =\n | HostGuestEvent<\"beforeload\">\n | HostGuestEvent<\"load\">\n | HostEvent<\"beforeunload\">\n | HostEvent<\"unload\">\n | HostEventLoadAllGuests\n | HostEventContextChange\n | HostEventError;\n\n/** @public */\nexport type InstalledExtensions = Record<Extension[\"id\"], Extension[\"url\"]>;\n/** @public */\nexport type ExtensionsProvider = () => Promise<InstalledExtensions>;\n\n/**\n * Values for shared context. Must be a plain object, serializable to JSON.\n * @public\n */\nexport type SharedContextValues = Record<string, unknown>;\n\n/** @public */\nexport interface HostConfig {\n /**\n * Human-readable \"slug\" name of the extensible area--often an entire app.\n * This string serves as a namespace for extension points within the area.\n */\n hostName: string;\n /**\n * A DOM element _outside_ of the React root. This is necessary to preserve\n * the lifetime of the iframes which are running extension objects; if they\n * live inside the React root, then React could unexpectedly re-render the\n * iframe tags themselves at any time, causing a reload of the frame.\n */\n runtimeContainer?: HTMLElement;\n /**\n * Copiously log lifecycle events.\n */\n debug?: boolean;\n /**\n * Default options to use for every guest Port.\n *\n * If `config.debug` is true, then the guest options will have `debug: true`\n * unless `debug: false` is explicitly passed in `guestOptions`.\n */\n guestOptions?: PortOptions;\n /**\n * A read-only dictionary of values that the host will supply to all the\n * guests.\n */\n sharedContext?: SharedContextValues;\n}\n\n/**\n * Callback to use to filter the list returned from {@link Host.(getLoadedGuests:2)}\n * @public\n */\ntype GuestFilter = (item: GuestEmitter) => boolean;\n\nconst passAllGuests = () => true;\n\n/**\n * Manager object for connecting to {@link @adobe/uix-guest#GuestServer |\n * GuestServers} and {@link @adobe/uix-guest#GuestUI | GuestUIs}, providing and\n * receiving their APIs, and providing them to the app for interacting with UI.\n *\n * @remarks\n * The Host object is the main connection manager for all UIX Guests.\n * Making an app extensible requires creating a Host object.\n *\n * The extensible app using the Hostis responsible for providing a list of\n * extension references to the Host object. Use {@link\n * createExtensionRegistryProvider} for that purpose. Once you have retrieved a\n * list of extensions available to the host app, pass it to {@link Host.load}.\n *\n * When a Host creates a Guest, it must create an `<iframe>` element to contain\n * the Guest's main {@link @adobe/uix-guest#GuestServer} runtime, which runs\n * invisibly in the background. To do this, the Host creates a hidden container\n * in the body of the document. It is a `<div>` element with the attribute\n * `data-uix-guest-container`. Loaded GuestServers will be injected into this\n * hidden element and removed as necessary. When {@link Host.unload} is called,\n * the Host removes the hidden container from the document after unloading.\n *\n * @public\n */\nexport class Host extends Emitter<HostEvents> {\n /**\n * {@inheritDoc HostEventLoadAllGuests}\n * @eventProperty\n */\n public loadallguests: HostEventLoadAllGuests;\n\n /**\n * One guest has loaded.\n * @eventProperty\n */\n public guestload: HostGuestEvent<\"load\">;\n\n /**\n * About to attempt to load and connect to a Guest.\n * @eventProperty\n */\n public guestbeforeload: HostGuestEvent<\"beforeload\">;\n\n /**\n * About to unload a guest and remove its {@link @adobe/uix-guest#GuestServer}\n * instance as well as all its {@link @adobe/uix-guest#GuestUI} instances.\n * @eventProperty\n */\n public guestbeforeunload: HostGuestEvent<\"beforeunload\">;\n\n /**\n * Unloaded a guest and removed its {@link @adobe/uix-guest#GuestServer}\n * instance as well as all its {@link @adobe/uix-guest#GuestUI} instances.\n * @eventProperty\n */\n public guestunload: HostGuestEvent<\"unload\">;\n\n /**\n * {@inheritDoc HostEventContextChange}\n * @eventProperty\n */\n public contextchange: HostEventContextChange;\n\n /**\n * {@inheritDoc HostEventError}\n * @eventProperty\n */\n public error: HostEventError;\n\n private static containerStyle = {\n position: \"fixed\",\n width: \"1px\",\n height: \"1px\",\n pointerEvents: \"none\",\n opacity: 0,\n top: 0,\n left: \"-1px\",\n };\n /**\n * Unique string identifying the Host object.\n */\n hostName: string;\n /**\n * `true` if any extension in {@link Host.guests} has created a {@link\n * @adobe/uix-guest#GuestServer}, but the Guest has not yet loaded.\n */\n loading = false;\n /**\n * A Map of of the loaded guests.\n */\n guests: PortMap = new Map();\n private cachedCapabilityLists: WeakMap<object, Port[]> = new WeakMap();\n private runtimeContainer: HTMLElement;\n private guestOptions: PortOptions;\n private logger: Console = quietConsole;\n private sharedContext: SharedContextValues;\n constructor(config: HostConfig) {\n super(config.hostName);\n const { guestOptions = {} } = config;\n this.guestOptions = {\n ...guestOptions,\n debug: guestOptions.debug === false ? false : !!config.debug,\n };\n this.hostName = config.hostName;\n this.sharedContext = config.sharedContext || {};\n this.runtimeContainer = config.runtimeContainer;\n if (config.debug) {\n this.logger = debugHost(this);\n }\n }\n /**\n * Return all loaded guests.\n */\n getLoadedGuests<T = unknown>(): Port<T>[];\n /**\n * Return loaded guests which satisfy the passed test function.\n */\n getLoadedGuests<T = unknown>(filter: GuestFilter): Port<T>[];\n /**\n * Return loaded guests which expose the provided {@link CapabilitySpec}.\n */\n getLoadedGuests<Apis extends GuestApis>(\n capabilities: CapabilitySpec<Apis>\n ): Port<GuestApis>[];\n getLoadedGuests<Apis extends GuestApis = never>(\n filterOrCapabilities?: CapabilitySpec<Apis> | GuestFilter\n ): Port<GuestApis>[] {\n if (typeof filterOrCapabilities === \"object\") {\n return this.getLoadedGuestsWith<Apis>(filterOrCapabilities);\n }\n const filter = filterOrCapabilities || passAllGuests;\n const result = [];\n for (const guest of this.guests.values()) {\n if (guest.isReady() && filter(guest)) {\n result.push(guest as Port<GuestApis>);\n }\n }\n return result;\n }\n /**\n * Set the object of shared values that all Guests can access via {@link @adobe/uix-guest#GuestServer.sharedContext}.\n * This overwrites any previous object.\n *\n * @example Exposes `authToken` to all Guests. Guests can call `this.sharedContext.get('authToken')` to retrieve this value.\n * ```javascript\n * host.shareContext({\n * authToken: '82ba19b'\n * });\n * ```\n *\n * @example Overwrites the previous sharedContext, deleting `authToken` and providing `secret` and `auth` instead.\n * ```javascript\n * host.shareContext({\n * secret: 'squirrel',\n * auth: false\n * });\n * ```\n */\n shareContext(context: SharedContextValues): void;\n /**\n * Update the object of shared values that all Guests can access via {@link\n * @adobe/uix-guest#GuestServer.sharedContext}. This method takes a callback\n * which receives the previous context and may return an entirely new context,\n * or new values merged with the old context.\n *\n * @remarks This callback pattern allows the shared context values to be\n * mutable while the internal context object references are immutable, which\n * is important for synchronizing. with guests.\n *\n * @example Overwrites a context object based on the previous one.\n * ```javascript\n * host.shareContext(oldContext => ({\n * counter: oldContext.counter + 1\n * }))\n * ```\n *\n * @example Updates a context while preserving other existing values.\n * ```javascript\n * host.shareContext(oldContext => ({\n * ...oldContext,\n * counter: oldContext.counter + 1\n * }))\n * ```\n */\n shareContext(\n setter: (context: SharedContextValues) => SharedContextValues\n ): void;\n shareContext(\n setter: (context: SharedContextValues) => SharedContextValues\n ): void;\n shareContext(\n setterOrContext:\n | ((context: SharedContextValues) => SharedContextValues)\n | SharedContextValues\n ) {\n if (typeof setterOrContext === \"function\") {\n this.sharedContext = setterOrContext(this.sharedContext);\n } else {\n this.sharedContext = setterOrContext;\n }\n this.emit(\"contextchange\", {\n host: this,\n context: this.sharedContext,\n });\n }\n /**\n * Load extension into host application from provided extension description.\n * Returned promise resolves when all extensions are loaded and registered.\n *\n * @param extensions - List of extension descriptors. Normally, the Host should receive this value from an {@link ExtensionsProvider}.\n * @param options - Custom options to be used as defaults for each {@link Port} object created for each guest.\n * @returns Promise which resolves when all guests have been loaded.\n */\n async load(\n extensions: InstalledExtensions,\n options?: PortOptions\n ): Promise<void> {\n this.runtimeContainer =\n this.runtimeContainer || this.createRuntimeContainer(window);\n const failed: Port[] = [];\n const loaded: Port[] = [];\n this.loading = true;\n await Promise.all(\n Object.entries(extensions).map(async ([id, url]) => {\n const port = await this.loadOneGuest(id, url, options);\n (port.error ? failed : loaded).push(port);\n })\n );\n this.loading = false;\n this.emit(\"loadallguests\", { host: this, failed, loaded });\n }\n /**\n * Unload all extensions and remove their frames/workers. Use this to unmount\n * a UI or when switching to a different extensible UI.\n */\n async unload(): Promise<void> {\n this.emit(\"beforeunload\", { host: this });\n await Promise.all([...this.guests.values()].map((guest) => guest.unload()));\n this.guests.clear();\n this.runtimeContainer.parentElement.removeChild(this.runtimeContainer);\n this.emit(\"unload\", { host: this });\n }\n private createRuntimeContainer(window: Window) {\n const { document } = window;\n const container = document.createElement(\"div\");\n container.setAttribute(\"data-uix-guest-container\", this.hostName);\n container.setAttribute(\"role\", \"presentation\");\n container.setAttribute(\"aria-hidden\", \"true\");\n Object.assign(container.style, Host.containerStyle);\n document.body.appendChild(container);\n return container;\n }\n private async loadOneGuest<T = unknown>(\n id: string,\n urlString: string,\n options: PortOptions = {}\n ): Promise<Port<T>> {\n let guest = this.guests.get(id);\n if (!guest) {\n const url = new URL(urlString);\n guest = new Port({\n owner: this.hostName,\n id,\n url,\n runtimeContainer: this.runtimeContainer,\n options: {\n ...this.guestOptions,\n ...options,\n },\n logger: this.logger,\n sharedContext: this.sharedContext,\n events: this as Emits,\n });\n this.guests.set(id, guest);\n }\n this.emit(\"guestbeforeload\", { guest, host: this });\n try {\n await guest.load();\n } catch (e: unknown) {\n const error = new Error(\n `Guest ${guest.id} failed to load: at ${guest.url}: ${\n e instanceof Error ? e.stack : String(e)\n }`\n );\n this.emit(\"error\", { host: this, guest, error });\n return guest;\n }\n // this new guest might have new capabilities, so the identities of the\n // cached capability sets will need to change, to alert subscribers\n this.cachedCapabilityLists = new WeakMap();\n this.emit(\"guestload\", { guest, host: this });\n return guest;\n }\n private getLoadedGuestsWith<Apis extends GuestApis>(\n capabilities: CapabilitySpec<Apis>\n ) {\n if (this.cachedCapabilityLists.has(capabilities)) {\n return this.cachedCapabilityLists.get(capabilities);\n }\n const guestsWithCapabilities = this.getLoadedGuests((guest) =>\n guest.hasCapabilities(capabilities)\n );\n this.cachedCapabilityLists.set(capabilities, guestsWithCapabilities);\n return guestsWithCapabilities;\n }\n}\n","/*\nCopyright 2022 Adobe. All rights reserved.\nThis file is licensed to you under the Apache License, Version 2.0 (the \"License\");\nyou may not use this file except in compliance with the License. You may obtain a copy\nof the License at http://www.apache.org/licenses/LICENSE-2.0\n\nUnless required by applicable law or agreed to in writing, software distributed under\nthe License is distributed on an \"AS IS\" BASIS, WITHOUT WARRANTIES OR REPRESENTATIONS\nOF ANY KIND, either express or implied. See the License for the specific language\ngoverning permissions and limitations under the License.\n*/\n\nimport type {\n Emits,\n GuestConnection,\n GuestConnectionEvents,\n HostMethodAddress,\n RemoteHostApis,\n GuestApis,\n CrossRealmObject,\n Unsubscriber,\n VirtualApi,\n UIHostMethods,\n} from \"@adobe/uix-core\";\nimport {\n Emitter,\n connectIframe,\n formatHostMethodAddress,\n} from \"@adobe/uix-core\";\nimport { normalizeIframe } from \"./dom-utils\";\n\n/**\n * A specifier for methods to be expected on a remote interface.\n *\n * @remarks\n * A CapabilitySpec is a description of an interface, like a very simplified\n * type definition. It specifies an object structure and the paths in that\n * structure that must be functions. (It doesn't specify anything about the\n * signatures or return values of those functions.)\n *\n * Use CapabilitySpec objects as queries, or filters, to get a subset of\n * installed extensions which have registered methods which match the spec.\n *\n * @example\n * As an extensible app developer, you are making an extension point for spell\n * check. Your code expects extensions to register an API `spellCheck` with\n * methods called `spellCheck.correct(text)` and `spellCheck.suggest(text)`.\n *\n * ```javascript\n * async function correctText(text) {\n * const spellCheckers = host.getLoadedGuests({\n * spellCheck: [\n * 'correct',\n * 'suggest'\n * ]\n * });\n * let correcting = text;\n * for (const checker of spellCheckers) {\n * correcting = await checker.apis.spellCheck.correct(correcting);\n * }\n * return Promise.all(checkers.map(checker =>\n * checker.apis.spellCheck.suggest(correcting)\n * ));\n * }\n * ```\n *\n * @public\n */\nexport type CapabilitySpec<T extends GuestApis> = {\n [Name in keyof T]: (keyof T[Name])[];\n};\n\n/**\n * Interface for decoupling of guest Penpal object\n * @internal\n */\ninterface GuestProxyWrapper {\n // #region Properties (1)\n\n /**\n * Methods from guest\n */\n apis: RemoteHostApis;\n\n // #endregion Properties (1)\n\n // #region Public Methods (1)\n\n /**\n * Emit an event in the guest frame\n */\n emit(type: string, detail: unknown): Promise<void>;\n\n // #endregion Public Methods (1)\n}\n\n/** @public */\nexport type PortOptions = {\n /**\n * Time in milliseconds to wait for the guest to connect before throwing.\n */\n timeout?: number;\n /**\n * Set true to log copiously in the console.\n */\n debug?: boolean;\n};\n\nconst defaultOptions = {\n timeout: 10000,\n debug: false,\n};\n\n/**\n * A Port is the Host-maintained object representing an extension running as a\n * guest. It exposes methods registered by the Guest, and can provide Host\n * methods back to the guest.\n *\n * @remarks\n * When the Host object loads extensions via {@link Host.load}, it creates a\n * Port object for each extension. When retrieving and filtering extensions\n * via {@link Host.(getLoadedGuests:2)}, a list of Port objects is returned. From\n * the point of view of the extensible app using the Host object, extensions\n * are always Port objects, which expose the methods registered by the\n * extension at the {@link Port.apis} property.\n *\n * @privateRemarks\n * We've gone through several possible names for this object. GuestProxy,\n * GuestInterface, GuestConnection, etc. \"Port\" is not ideal, but it conflicted\n * the least with other types we defined in early drafts. It's definitely\n * something we should review.\n * @public\n */\nexport class Port<GuestApi = unknown>\n extends Emitter<GuestConnectionEvents>\n implements GuestConnection\n{\n public get apis() {\n if (this.isReady() && this.guestServer) {\n const server = this.guestServer.getRemoteApi();\n return server && server.apis;\n }\n }\n\n // #region Properties (13)\n\n private debug: boolean;\n private logger?: Console;\n private guestServerFrame: HTMLIFrameElement;\n private hostApis: RemoteHostApis = {};\n private isLoaded = false;\n private runtimeContainer: HTMLElement;\n private sharedContext: Record<string, unknown>;\n private subscriptions: Unsubscriber[] = [];\n private timeout: number;\n\n /**\n * Dictionary of namespaced methods that were registered by this guest at the\n * time of connection, using {@link @adobe/uix-guest#register}.\n *\n * @remarks\n * These methods are proxy methods; you can only pass serializable objects to\n * them, not class instances, methods or callbacks.\n * @public\n */\n /**\n * If any errors occurred during the loading of guests, this property will\n * contain the error that was raised.\n * @public\n */\n error?: Error;\n /**\n * The URL of the guest provided by the extension registry. The Host will\n * load this URL in the background, in the invisible the bootstrap frame, so\n * this URL must point to a page that calls {@link @adobe/uix-guest#register}\n * when it loads.\n */\n public url: URL;\n private guestServer: CrossRealmObject<GuestProxyWrapper>;\n\n // #endregion Properties (13)\n\n // #region Constructors (1)\n\n constructor(config: {\n owner: string;\n id: string;\n url: URL;\n /**\n * An alternate DOM element to use for invisible iframes. Will create its\n * own if this option is not populated with a DOM element.\n */\n runtimeContainer: HTMLElement;\n options: PortOptions;\n logger?: Console;\n /**\n * Initial object to populate the shared context with. Once the guest\n * connects, it will be able to access these properties.\n */\n sharedContext: Record<string, unknown>;\n events: Emits;\n }) {\n super(config.id);\n const { timeout, debug } = { ...defaultOptions, ...(config.options || {}) };\n this.timeout = timeout;\n this.debug = debug;\n this.logger = config.logger;\n this.id = config.id;\n this.url = config.url;\n this.runtimeContainer = config.runtimeContainer;\n this.sharedContext = config.sharedContext;\n this.subscriptions.push(\n config.events.addEventListener(\"contextchange\", async (event) => {\n this.sharedContext = (\n (event as CustomEvent).detail as unknown as Record<string, unknown>\n ).context as Record<string, unknown>;\n await this.load();\n await this.guestServer\n .getRemoteApi()\n .emit(\"contextchange\", { context: this.sharedContext });\n })\n );\n }\n\n // #endregion Constructors (1)\n\n // #region Public Methods (6)\n\n /**\n * Connect an iframe element which is displaying another page in the extension\n * with the extension's bootstrap frame, so they can share context and events.\n */\n public attachUI<T = unknown>(\n iframe: HTMLIFrameElement\n ): Promise<CrossRealmObject<T>> {\n return this.attachFrame(iframe, {\n onIframeResize: (dimensions: { height: number; width: number }) => {\n this.emit(\"guestresize\", {\n dimensions,\n guestPort: this,\n iframe: iframe,\n });\n },\n } as UIHostMethods);\n }\n\n /**\n * Returns true if the guest has registered methods matching the provided\n * capability spec. A capability spec is simply an object whose properties are\n * declared in an array of keys, description the names of the functions and\n * methods that the Port will expose.\n */\n public hasCapabilities(requiredMethods: CapabilitySpec<GuestApis>) {\n this.assertReady();\n return Object.keys(requiredMethods).every((key) => {\n if (!Reflect.has(this.apis, key)) {\n return false;\n }\n const api = this.apis[key];\n const methodList = requiredMethods[\n key as keyof typeof requiredMethods\n ] as string[];\n return methodList.every(\n (methodName: string) =>\n Reflect.has(api, methodName) &&\n typeof api[methodName as keyof typeof api] === \"function\"\n );\n });\n }\n\n /**\n * True when al extensions have loaded.\n */\n public isReady(): boolean {\n return this.isLoaded && !this.error;\n }\n\n /**\n * Loads the extension. Returns a promise which resolves when the extension\n * has loaded. The Host calls this method after retrieving extensions.\n */\n public async load() {\n try {\n if (!this.apis) {\n await this.connect();\n }\n return this.apis;\n } catch (e) {\n this.guestServer = null;\n this.error = e instanceof Error ? e : new Error(String(e));\n throw e;\n }\n }\n\n /**\n * The host-side equivalent of {@link @adobe/uix-guest#register}. Pass a set\n * of methods down to the guest as proxies.\n * Merges at the first level, the API level. Overwrites a deeper levels.\n */\n public provide(apis: RemoteHostApis) {\n for (const [apiNamespace, methods] of Object.entries(apis)) {\n this.hostApis[apiNamespace] = this.hostApis[apiNamespace] || {};\n Object.assign(this.hostApis[apiNamespace], methods);\n }\n this.emit(\"hostprovide\", { guestPort: this, apis });\n }\n\n /**\n * Disconnect from the extension.\n */\n public async unload(): Promise<void> {\n if (this.guestServerFrame && this.guestServerFrame.parentElement) {\n this.guestServerFrame.parentElement.removeChild(this.guestServerFrame);\n this.guestServerFrame = undefined;\n }\n this.emit(\"unload\", { guestPort: this });\n }\n\n // #endregion Public Methods (6)\n\n // #region Private Methods (6)\n\n private assert(\n condition: boolean,\n errorMessage: () => string\n ): asserts condition {\n if (!condition) {\n throw new Error(\n `Error in guest extension \"${this.id}\": ${errorMessage()}`\n );\n }\n }\n\n private assertReady() {\n this.assert(this.isReady(), () => \"Attempted to interact before loaded\");\n }\n\n private attachFrame<T = unknown>(\n iframe: HTMLIFrameElement,\n addedMethods: object = {}\n ) {\n // at least this is necessary\n normalizeIframe(iframe);\n this.logger.log(\"attachFrame\", iframe);\n return connectIframe<T>(\n iframe,\n {\n logger: this.logger,\n targetOrigin: this.url.origin,\n timeout: this.timeout,\n },\n {\n getSharedContext: () => this.sharedContext,\n invokeHostMethod: (address: HostMethodAddress) =>\n this.invokeHostMethod(address),\n ...addedMethods,\n }\n );\n }\n\n private async connect() {\n const serverFrame =\n this.runtimeContainer.ownerDocument.createElement(\"iframe\");\n normalizeIframe(serverFrame);\n serverFrame.setAttribute(\"aria-hidden\", \"true\");\n serverFrame.setAttribute(\"src\", this.url.href);\n this.guestServerFrame = serverFrame;\n this.runtimeContainer.appendChild(serverFrame);\n if (this.logger) {\n this.logger.info(\n `Guest ${this.id} attached iframe of ${this.url.href}`,\n this\n );\n }\n this.guestServer = await this.attachFrame<GuestProxyWrapper>(serverFrame);\n this.isLoaded = true;\n if (this.logger) {\n this.logger.info(\n `Guest ${this.id} established connection, received methods`,\n this.apis,\n this\n );\n }\n }\n\n private getHostMethodCallee<T = unknown>(\n { name, path }: HostMethodAddress,\n methodSource: RemoteHostApis\n ): RemoteHostApis<VirtualApi> {\n const dots = (level: number) => `host.${path.slice(0, level).join(\".\")}`;\n const methodCallee = path.reduce((current, prop, level) => {\n this.assert(\n Reflect.has(current, prop),\n () => `${dots(level)} has no property \"${prop}\"`\n );\n const next = current[prop];\n this.assert(\n typeof next === \"object\",\n () =>\n `${dots(\n level\n )}.${prop} is not an object; namespaces must be objects with methods`\n );\n return next as RemoteHostApis<GuestApi>;\n }, methodSource);\n this.assert(\n typeof methodCallee[name] === \"function\" &&\n Reflect.has(methodCallee, name),\n () => `\"${dots(path.length - 1)}.${name}\" is not a function`\n );\n return methodCallee;\n }\n\n private invokeHostMethod<T = unknown>(\n address: HostMethodAddress,\n privateMethods?: RemoteHostApis\n ): T {\n const { name, path, args = [] } = address;\n this.assert(name && typeof name === \"string\", () => \"Method name required\");\n this.assert(\n path.length > 0,\n () =>\n `Cannot call a method directly on the host; \".${name}()\" must be in a namespace.`\n );\n let methodCallee;\n if (privateMethods) {\n try {\n methodCallee = this.getHostMethodCallee(address, privateMethods);\n } catch (e) {\n this.logger.warn(\n `Private method ${formatHostMethodAddress(address)} not found!`,\n address\n );\n }\n }\n if (!methodCallee) {\n methodCallee = this.getHostMethodCallee(address, this.hostApis);\n }\n const method = methodCallee[name] as (...args: unknown[]) => T;\n this.emit(\"beforecallhostmethod\", { guestPort: this, name, path, args });\n return method.apply(methodCallee, [\n { id: this.id, url: this.url },\n ...args,\n ]) as T;\n }\n\n // #endregion Private Methods (6)\n}\n","/**\nCopyright 2022 Adobe. All rights reserved.\nThis file is licensed to you under the Apache License, Version 2.0 (the \"License\");\nyou may not use this file except in compliance with the License. You may obtain a copy\nof the License at http://www.apache.org/licenses/LICENSE-2.0\n\nUnless required by applicable law or agreed to in writing, software distributed under\nthe License is distributed on an \"AS IS\" BASIS, WITHOUT WARRANTIES OR REPRESENTATIONS\nOF ANY KIND, either express or implied. See the License for the specific language\ngoverning permissions and limitations under the License.\n*/\n\n/**\n * Strings to be used as values in a space-separated HTML attribute.\n * @internal\n */\nexport type AttrTokens<T> = string | T[];\n\n/**\n * Normalize an argument that may be either a space-separated HTML attribute\n * value or an array of those attribute values into an array of those values.\n * @internal\n */\nexport const tokenizeAttrValues = <T>(tokens: AttrTokens<T>) =>\n (typeof tokens === \"string\" ? tokens.split(\" \") : tokens) as T[];\n\n/**\n * Merge, deduplicate and typecheck a set of strings for an HTML attribute that\n * requires those values to be space-separated tokens.\n * @internal\n *\n * @remarks\n * Useful as a typed interface for any DOM property that is a DOMTokenList.\n * While the DOMTokenList does its own deduplication, it's slightly slower,\n * and while it may do validation of legal tokens, it only does so at runtime.\n * Using {@link AttrTokens} and this function adds typing and autocomplete.\n *\n *\n * @example\n * ```typescript\n * type AllowedClasses =\n * | \"primary\"\n * | \"light\"\n * | \"large\"\n * | \"quiet\";\n *\n * // combine with existing classes and set attribute directly\n * function setStyles(elm: HTMLElement, styles: AttrTokens<AllowedClasses>) {\n * const classNames = mergeAttrValues(\n * elm.className as AttrTokens<AllowedClasses>,\n * styles\n * );\n * elm.className = classNames.join(' ');\n * }\n *\n * // use DOM property directly, but now it's typed!\n * function setSandbox(\n * iframe: HTMLIframeElement,\n * allowedSandboxes: AttrTokens<\"allow-scripts\" | \"allow-popups\">\n * ) {\n * mergeAttrValues(allowedSandboxes).forEach(val => iframe.sandbox.add(val));\n * }\n * ```\n */\nexport const mergeAttrValues = <T>(...tokenLists: AttrTokens<T>[]) => {\n const allMerged = new Set<T>();\n for (const tokenList of tokenLists) {\n for (const token of tokenizeAttrValues(tokenList)) {\n allMerged.add(token);\n }\n }\n return [...allMerged];\n};\n","/**\nCopyright 2022 Adobe. All rights reserved.\nThis file is licensed to you under the Apache License, Version 2.0 (the \"License\");\nyou may not use this file except in compliance with the License. You may obtain a copy\nof the License at http://www.apache.org/licenses/LICENSE-2.0\n\nUnless required by applicable law or agreed to in writing, software distributed under\nthe License is distributed on an \"AS IS\" BASIS, WITHOUT WARRANTIES OR REPRESENTATIONS\nOF ANY KIND, either express or implied. See the License for the specific language\ngoverning permissions and limitations under the License.\n*/\n\nimport { HTMLAttributeReferrerPolicy } from \"react\";\nimport { AttrTokens, mergeAttrValues } from \"./attribute-normalizers\";\n\n/**\n * Sandbox permissions that guest iframes are allowed to have.\n * @internal\n */\ntype SandboxPermission =\n | \"downloads\"\n | \"orientation-lock\"\n | \"pointer-lock\"\n | \"popups\"\n | \"presentation\"\n | \"same-origin\"\n | \"scripts\"\n | \"storage-access-by-user-activation\"\n | \"top-navigation-by-user-activation\";\n\n/**\n * Character strings allowed in \"sandbox=\" attribute.\n * @internal\n */\nexport type SandboxToken = `allow-${SandboxPermission}`;\n\n/**\n * Merge lists of attrs together\n * @internal\n */\nexport const makeSandboxAttrs = (...sandboxes: AttrTokens<SandboxToken>[]) =>\n mergeAttrValues<SandboxToken>(...sandboxes);\n\n/**\n * Limit provided set of \"sandbox\" attributes to a list of legal ones.\n * @internal\n */\nexport const requiredIframeProps = {\n // must not require this until app builder supports CSP\n // csp: \"frame-ancestors 'self'\",\n \"data-uix-guest\": \"true\",\n role: \"presentation\",\n referrerPolicy: \"strict-origin\" as HTMLAttributeReferrerPolicy,\n};\n\nconst requiredIframeAttrEntries = Object.entries(requiredIframeProps);\n\n/**\n * Set and/or verify the required properties for use with uix-sdk on an iframe.\n * @internal\n */\nexport const normalizeIframe = (iframe: HTMLIFrameElement) => {\n for (const [attr, value] of requiredIframeAttrEntries) {\n iframe.setAttribute(attr, value);\n }\n};\n","/*\nCopyright 2022 Adobe. All rights reserved.\nThis file is licensed to you under the Apache License, Version 2.0 (the \"License\");\nyou may not use this file except in compliance with the License. You may obtain a copy\nof the License at http://www.apache.org/licenses/LICENSE-2.0\n\nUnless required by applicable law or agreed to in writing, software distributed under\nthe License is distributed on an \"AS IS\" BASIS, WITHOUT WARRANTIES OR REPRESENTATIONS\nOF ANY KIND, either express or implied. See the License for the specific language\ngoverning permissions and limitations under the License.\n*/\n\n/* eslint-disable @typescript-eslint/restrict-template-expressions */\n/* eslint-disable @typescript-eslint/no-unsafe-member-access */\n/**\n * Adapter to attach console logging listeners to a Host running in an app\n * @hidden\n */\nimport {\n debugEmitter,\n EmitterDebugLogger,\n Emits,\n GuestEmitter,\n} from \"@adobe/uix-core\";\nimport type { HostEventLoadAllGuests, HostEvents } from \"./host.js\";\n\nexport function debugHost(host: Emits<HostEvents>): EmitterDebugLogger {\n const hostLogger = debugEmitter(host, {\n theme: \"blue medium\",\n type: \"Host\",\n });\n hostLogger\n .listen(\"guestbeforeload\", (log, event) => {\n const { detail } = event;\n const guest = detail.guest as GuestEmitter;\n log.info(event, `Guest ID ${guest.id}`);\n const portLogger = debugEmitter(guest, {\n theme: \"green medium\",\n type: \"Port\",\n id: `${host.id} ➔ ${guest.id}`,\n });\n portLogger\n .listen(\"hostprovide\", (log, event) => {\n log.info(\"received APIs\", event.detail.apis);\n })\n .listen(\"beforecallhostmethod\", (log, event) => {\n log.info(event.detail);\n })\n .listen(\"guestresize\", (log, event) => {\n log.info(event.detail);\n })\n .listen(\"unload\", (log, event) => {\n log.info(event.detail);\n log.detach();\n });\n })\n .listen(\"guestload\", (log, e) => {\n log.info(e.detail.guest.id, e.detail.guest);\n })\n .listen(\"error\", (log, e) => {\n log.error(`Error: ${e.detail.error.message}`, e);\n })\n .listen(\n \"loadallguests\",\n (log, { detail: { failed, loaded, host } }: HostEventLoadAllGuests) => {\n if (failed.length > 0) {\n log.error(\"%d guests failed to load!\", failed.length);\n }\n log.info(\"%d guests loaded\", loaded.length, host);\n }\n )\n .listen(\"unload\", (log) => {\n log.info(\"Unloaded guest and container.\");\n log.detach();\n });\n return hostLogger;\n}\n","/*\nCopyright 2022 Adobe. All rights reserved.\nThis file is licensed to you under the Apache License, Version 2.0 (the \"License\");\nyou may not use this file except in compliance with the License. You may obtain a copy\nof the License at http://www.apache.org/licenses/LICENSE-2.0\n\nUnless required by applicable law or agreed to in writing, software distributed under\nthe License is distributed on an \"AS IS\" BASIS, WITHOUT WARRANTIES OR REPRESENTATIONS\nOF ANY KIND, either express or implied. See the License for the specific language\ngoverning permissions and limitations under the License.\n*/\n\nimport { InstalledExtensions, ExtensionsProvider } from \"../host.js\";\n\ninterface ExtensionDefinition {\n name: string;\n title: string;\n description: string;\n icon: string;\n publisher: string;\n endpoints: Record<string, EndpointDefinition>;\n xrInfo: ExtensionInfo;\n status: string;\n}\n\ntype EndpointDefinition = Record<string, Array<OperationDefinition>>;\n\ninterface ExtensionInfo {\n supportEmail: string;\n appId: string;\n}\n\ninterface OperationDefinition {\n href: string;\n metadata: OperationMetadata;\n}\n\ninterface OperationMetadata {\n services: Array<object>;\n profile: OperationProfile;\n}\n\ninterface OperationProfile {\n client_id: string;\n scope: string;\n}\n\n/** @public */\nexport interface ExtensionRegistryEndpointRegistration {\n service: string;\n extensionPoint: string;\n version: string;\n}\n\n/** @public */\nexport interface ExtensionRegistryExtensionRegistration\n extends ExtensionRegistryEndpointRegistration {\n imsOrg: string;\n}\n\n/** @public */\nexport interface ExtensionRegistryConnection {\n baseUrl?: string;\n apiKey: string;\n auth: {\n schema: \"Basic\" | \"Bearer\";\n imsToken: string;\n };\n}\n\n/** @public */\nexport interface ExtensionRegistryConfig\n extends ExtensionRegistryExtensionRegistration,\n ExtensionRegistryConnection {}\n\nfunction buildEndpointPath(\n config: ExtensionRegistryEndpointRegistration\n): string {\n return `${config.service}/${config.extensionPoint}/${config.version}`;\n}\n\nfunction ensureProtocolSpecified(url: string) {\n if (url.startsWith(\"https://\")) {\n return url;\n }\n if (url.startsWith(\"http://\")) {\n return url;\n }\n return `https://${url}`;\n}\n\nasync function fetchExtensionsFromRegistry(\n config: ExtensionRegistryConfig\n): Promise<Array<ExtensionDefinition>> {\n const resp = await fetch(\n `${ensureProtocolSpecified(\n config.baseUrl || \"appregistry.adobe.io\"\n )}/myxchng/v1/org/${encodeURIComponent(\n config.imsOrg\n )}/xtn/${buildEndpointPath(config)}?auth=true`,\n {\n headers: {\n Accept: \"application/json\",\n Authorization: `${config.auth.schema} ${config.auth.imsToken}`, // todo: check if auth schema needed (initial implementation was without it)\n \"X-Api-Key\": config.apiKey,\n },\n }\n );\n\n if (resp.status != 200) {\n throw new Error(\n `extension registry returned non-200 response (${\n resp.status\n }): ${await resp.text()}`\n );\n }\n\n return await resp.json();\n}\n\nfunction extensionRegistryExtensionsProvider(\n config: ExtensionRegistryConfig\n): Promise<InstalledExtensions> {\n const erEndpoint = buildEndpointPath(config);\n return fetchExtensionsFromRegistry(config).then((out) =>\n out.reduce((a, e: ExtensionDefinition) => {\n if (e.status !== \"PUBLISHED\") {\n return a;\n }\n\n return {\n ...a,\n // todo: make safer way to extract href\n [e.name]: e.endpoints[erEndpoint].view[0].href,\n };\n }, {})\n );\n\n return Promise.resolve({});\n}\n\n/**\n * Create a callback that fetches extensions from the registry.\n * @public\n */\nexport function createExtensionRegistryProvider(\n config: ExtensionRegistryConfig\n): ExtensionsProvider {\n return function () {\n return extensionRegistryExtensionsProvider(config);\n };\n}\n","/*\nCopyright 2022 Adobe. All rights reserved.\nThis file is licensed to you under the Apache License, Version 2.0 (the \"License\");\nyou may not use this file except in compliance with the License. You may obtain a copy\nof the License at http://www.apache.org/licenses/LICENSE-2.0\n\nUnless required by applicable law or agreed to in writing, software distributed under\nthe License is distributed on an \"AS IS\" BASIS, WITHOUT WARRANTIES OR REPRESENTATIONS\nOF ANY KIND, either express or implied. See the License for the specific language\ngoverning permissions and limitations under the License.\n*/\n\nimport { InstalledExtensions, ExtensionsProvider } from \"../host.js\";\n\n/**\n * Combine multiple {@link @adobe/uix-host#ExtensionsProvider} callbacks into a\n * single callback, which aggregates and dedupes all extensions from all\n * providers into one namespaced object.\n * @public\n */\nexport function combineExtensionsFromProviders(\n ...providers: Array<ExtensionsProvider>\n): ExtensionsProvider {\n return () =>\n Promise.all(providers.map((ep: ExtensionsProvider) => ep())).then(\n (extensionsBatches: Array<InstalledExtensions>) => {\n return Object.assign({}, ...extensionsBatches);\n }\n );\n}\n","/*\nCopyright 2022 Adobe. All rights reserved.\nThis file is licensed to you under the Apache License, Version 2.0 (the \"License\");\nyou may not use this file except in compliance with the License. You may obtain a copy\nof the License at http://www.apache.org/licenses/LICENSE-2.0\n\nUnless required by applicable law or agreed to in writing, software distributed under\nthe License is distributed on an \"AS IS\" BASIS, WITHOUT WARRANTIES OR REPRESENTATIONS\nOF ANY KIND, either express or implied. See the License for the specific language\ngoverning permissions and limitations under the License.\n*/\n\nimport { ExtensionsProvider } from \"../host.js\";\n\n/**\n * Mute any errors produced by provider.\n * This function would execute given provider and return its results as is, if any error occurs this provider will log it\n * any return an empty array of extensions.\n * @public\n */\nexport function mutedProvider(\n provider: ExtensionsProvider\n): ExtensionsProvider {\n return async () => {\n try {\n return await provider();\n } catch (error) {\n console.error(`Extension provider has failed: ${error.message}`, {\n error,\n });\n return {};\n }\n };\n}\n"]}
|
package/dist/port.d.ts
CHANGED
@@ -1,4 +1,4 @@
|
|
1
|
-
import type { Emits, GuestConnection,
|
1
|
+
import type { Emits, GuestConnection, GuestConnectionEvents, RemoteHostApis, GuestApis, CrossRealmObject } from "@adobe/uix-core";
|
2
2
|
import { Emitter } from "@adobe/uix-core";
|
3
3
|
/**
|
4
4
|
* A specifier for methods to be expected on a remote interface.
|
@@ -41,12 +41,6 @@ export declare type CapabilitySpec<T extends GuestApis> = {
|
|
41
41
|
[Name in keyof T]: (keyof T[Name])[];
|
42
42
|
};
|
43
43
|
/** @public */
|
44
|
-
declare type PortEvent<GuestApi, Type extends string = string, Detail = Record<string, unknown>> = NamedEvent<Type, Detail & Record<string, unknown> & {
|
45
|
-
guestPort: Port<GuestApi>;
|
46
|
-
}>;
|
47
|
-
/** @public */
|
48
|
-
export declare type PortEvents<GuestApi, HostApi extends Record<string, unknown> = Record<string, unknown>> = PortEvent<GuestApi, "hostprovide"> | PortEvent<GuestApi, "unload"> | PortEvent<GuestApi, "beforecallhostmethod", HostMethodAddress<HostApi>>;
|
49
|
-
/** @public */
|
50
44
|
export declare type PortOptions = {
|
51
45
|
/**
|
52
46
|
* Time in milliseconds to wait for the guest to connect before throwing.
|
@@ -77,7 +71,7 @@ export declare type PortOptions = {
|
|
77
71
|
* something we should review.
|
78
72
|
* @public
|
79
73
|
*/
|
80
|
-
export declare class Port<GuestApi> extends Emitter<
|
74
|
+
export declare class Port<GuestApi = unknown> extends Emitter<GuestConnectionEvents> implements GuestConnection {
|
81
75
|
get apis(): {
|
82
76
|
[x: string]: {};
|
83
77
|
};
|
@@ -135,7 +129,7 @@ export declare class Port<GuestApi> extends Emitter<PortEvents<GuestApi>> implem
|
|
135
129
|
* Connect an iframe element which is displaying another page in the extension
|
136
130
|
* with the extension's bootstrap frame, so they can share context and events.
|
137
131
|
*/
|
138
|
-
attachUI(iframe: HTMLIFrameElement): Promise<CrossRealmObject<
|
132
|
+
attachUI<T = unknown>(iframe: HTMLIFrameElement): Promise<CrossRealmObject<T>>;
|
139
133
|
/**
|
140
134
|
* Returns true if the guest has registered methods matching the provided
|
141
135
|
* capability spec. A capability spec is simply an object whose properties are
|
@@ -157,6 +151,7 @@ export declare class Port<GuestApi> extends Emitter<PortEvents<GuestApi>> implem
|
|
157
151
|
/**
|
158
152
|
* The host-side equivalent of {@link @adobe/uix-guest#register}. Pass a set
|
159
153
|
* of methods down to the guest as proxies.
|
154
|
+
* Merges at the first level, the API level. Overwrites a deeper levels.
|
160
155
|
*/
|
161
156
|
provide(apis: RemoteHostApis): void;
|
162
157
|
/**
|
@@ -170,5 +165,4 @@ export declare class Port<GuestApi> extends Emitter<PortEvents<GuestApi>> implem
|
|
170
165
|
private getHostMethodCallee;
|
171
166
|
private invokeHostMethod;
|
172
167
|
}
|
173
|
-
export {};
|
174
168
|
//# sourceMappingURL=port.d.ts.map
|
package/dist/port.d.ts.map
CHANGED
@@ -1 +1 @@
|
|
1
|
-
{"version":3,"file":"port.d.ts","sourceRoot":"","sources":["../src/port.ts"],"names":[],"mappings":"AAYA,OAAO,KAAK,EACV,KAAK,EACL,eAAe,EACf,
|
1
|
+
{"version":3,"file":"port.d.ts","sourceRoot":"","sources":["../src/port.ts"],"names":[],"mappings":"AAYA,OAAO,KAAK,EACV,KAAK,EACL,eAAe,EACf,qBAAqB,EAErB,cAAc,EACd,SAAS,EACT,gBAAgB,EAIjB,MAAM,iBAAiB,CAAC;AACzB,OAAO,EACL,OAAO,EAGR,MAAM,iBAAiB,CAAC;AAGzB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAoCG;AACH,oBAAY,cAAc,CAAC,CAAC,SAAS,SAAS,IAAI;KAC/C,IAAI,IAAI,MAAM,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC,IAAI,CAAC,CAAC,EAAE;CACrC,CAAC;AA0BF,cAAc;AACd,oBAAY,WAAW,GAAG;IACxB;;OAEG;IACH,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB;;OAEG;IACH,KAAK,CAAC,EAAE,OAAO,CAAC;CACjB,CAAC;AAOF;;;;;;;;;;;;;;;;;;;GAmBG;AACH,qBAAa,IAAI,CAAC,QAAQ,GAAG,OAAO,CAClC,SAAQ,OAAO,CAAC,qBAAqB,CACrC,YAAW,eAAe;IAE1B,IAAW,IAAI;;MAKd;IAID,OAAO,CAAC,KAAK,CAAU;IACvB,OAAO,CAAC,MAAM,CAAC,CAAU;IACzB,OAAO,CAAC,gBAAgB,CAAoB;IAC5C,OAAO,CAAC,QAAQ,CAAsB;IACtC,OAAO,CAAC,QAAQ,CAAS;IACzB,OAAO,CAAC,gBAAgB,CAAc;IACtC,OAAO,CAAC,aAAa,CAA0B;IAC/C,OAAO,CAAC,aAAa,CAAsB;IAC3C,OAAO,CAAC,OAAO,CAAS;IAExB;;;;;;;;OAQG;IACH;;;;OAIG;IACH,KAAK,CAAC,EAAE,KAAK,CAAC;IACd;;;;;OAKG;IACI,GAAG,EAAE,GAAG,CAAC;IAChB,OAAO,CAAC,WAAW,CAAsC;gBAM7C,MAAM,EAAE;QAClB,KAAK,EAAE,MAAM,CAAC;QACd,EAAE,EAAE,MAAM,CAAC;QACX,GAAG,EAAE,GAAG,CAAC;QACT;;;WAGG;QACH,gBAAgB,EAAE,WAAW,CAAC;QAC9B,OAAO,EAAE,WAAW,CAAC;QACrB,MAAM,CAAC,EAAE,OAAO,CAAC;QACjB;;;WAGG;QACH,aAAa,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;QACvC,MAAM,EAAE,KAAK,CAAC;KACf;IA2BD;;;OAGG;IACI,QAAQ,CAAC,CAAC,GAAG,OAAO,EACzB,MAAM,EAAE,iBAAiB,GACxB,OAAO,CAAC,gBAAgB,CAAC,CAAC,CAAC,CAAC;IAY/B;;;;;OAKG;IACI,eAAe,CAAC,eAAe,EAAE,cAAc,CAAC,SAAS,CAAC;IAkBjE;;OAEG;IACI,OAAO,IAAI,OAAO;IAIzB;;;OAGG;IACU,IAAI;;;IAajB;;;;OAIG;IACI,OAAO,CAAC,IAAI,EAAE,cAAc;IAQnC;;OAEG;IACU,MAAM,IAAI,OAAO,CAAC,IAAI,CAAC;IAYpC,OAAO,CAAC,MAAM;IAWd,OAAO,CAAC,WAAW;IAInB,OAAO,CAAC,WAAW;YAuBL,OAAO;IAyBrB,OAAO,CAAC,mBAAmB;IA4B3B,OAAO,CAAC,gBAAgB;CAkCzB"}
|
package/package.json
CHANGED
@@ -1,6 +1,6 @@
|
|
1
1
|
{
|
2
2
|
"name": "@adobe/uix-host",
|
3
|
-
"version": "0.
|
3
|
+
"version": "0.8.0",
|
4
4
|
"publishConfig": {
|
5
5
|
"access": "public"
|
6
6
|
},
|
@@ -18,7 +18,7 @@
|
|
18
18
|
],
|
19
19
|
"bugs": "https://github.com/adobe/uix-sdk/issues",
|
20
20
|
"dependencies": {
|
21
|
-
"@adobe/uix-core": "^0.
|
21
|
+
"@adobe/uix-core": "^0.8.0"
|
22
22
|
},
|
23
23
|
"files": [
|
24
24
|
"README.md",
|
package/src/debug-host.ts
CHANGED
@@ -20,15 +20,10 @@ import {
|
|
20
20
|
debugEmitter,
|
21
21
|
EmitterDebugLogger,
|
22
22
|
Emits,
|
23
|
-
|
23
|
+
GuestEmitter,
|
24
24
|
} from "@adobe/uix-core";
|
25
|
-
import type { PortEvents } from "./port.js";
|
26
25
|
import type { HostEventLoadAllGuests, HostEvents } from "./host.js";
|
27
26
|
|
28
|
-
type GenericPortEvents = PortEvents<Record<string, unknown>>;
|
29
|
-
|
30
|
-
type Portlike = GuestConnection & Emits<GenericPortEvents>;
|
31
|
-
|
32
27
|
export function debugHost(host: Emits<HostEvents>): EmitterDebugLogger {
|
33
28
|
const hostLogger = debugEmitter(host, {
|
34
29
|
theme: "blue medium",
|
@@ -37,7 +32,7 @@ export function debugHost(host: Emits<HostEvents>): EmitterDebugLogger {
|
|
37
32
|
hostLogger
|
38
33
|
.listen("guestbeforeload", (log, event) => {
|
39
34
|
const { detail } = event;
|
40
|
-
const guest = detail.guest as
|
35
|
+
const guest = detail.guest as GuestEmitter;
|
41
36
|
log.info(event, `Guest ID ${guest.id}`);
|
42
37
|
const portLogger = debugEmitter(guest, {
|
43
38
|
theme: "green medium",
|
@@ -51,6 +46,9 @@ export function debugHost(host: Emits<HostEvents>): EmitterDebugLogger {
|
|
51
46
|
.listen("beforecallhostmethod", (log, event) => {
|
52
47
|
log.info(event.detail);
|
53
48
|
})
|
49
|
+
.listen("guestresize", (log, event) => {
|
50
|
+
log.info(event.detail);
|
51
|
+
})
|
54
52
|
.listen("unload", (log, event) => {
|
55
53
|
log.info(event.detail);
|
56
54
|
log.detach();
|
package/src/host.ts
CHANGED
@@ -12,7 +12,7 @@ governing permissions and limitations under the License.
|
|
12
12
|
|
13
13
|
import type {
|
14
14
|
Extension,
|
15
|
-
|
15
|
+
GuestEmitter,
|
16
16
|
NamedEvent,
|
17
17
|
Emits,
|
18
18
|
GuestApis,
|
@@ -26,7 +26,7 @@ import { debugHost } from "./debug-host.js";
|
|
26
26
|
* Dictionary of {@link Port} objects by extension ID.
|
27
27
|
* @public
|
28
28
|
*/
|
29
|
-
export type PortMap = Map<string,
|
29
|
+
export type PortMap = Map<string, Port>;
|
30
30
|
|
31
31
|
/** @public */
|
32
32
|
export type HostEvent<
|
@@ -36,7 +36,7 @@ export type HostEvent<
|
|
36
36
|
/** @public */
|
37
37
|
type HostGuestEvent<Type extends string> = HostEvent<
|
38
38
|
`guest${Type}`,
|
39
|
-
{ guest:
|
39
|
+
{ guest: Port }
|
40
40
|
>;
|
41
41
|
|
42
42
|
/**
|
@@ -45,7 +45,7 @@ type HostGuestEvent<Type extends string> = HostEvent<
|
|
45
45
|
*/
|
46
46
|
export type HostEventLoadAllGuests = HostEvent<
|
47
47
|
"loadallguests",
|
48
|
-
{ failed:
|
48
|
+
{ failed: Port[]; loaded: Port[] }
|
49
49
|
>;
|
50
50
|
|
51
51
|
/**
|
@@ -121,7 +121,7 @@ export interface HostConfig {
|
|
121
121
|
* Callback to use to filter the list returned from {@link Host.(getLoadedGuests:2)}
|
122
122
|
* @public
|
123
123
|
*/
|
124
|
-
type GuestFilter = (item:
|
124
|
+
type GuestFilter = (item: GuestEmitter) => boolean;
|
125
125
|
|
126
126
|
const passAllGuests = () => true;
|
127
127
|
|
@@ -216,8 +216,7 @@ export class Host extends Emitter<HostEvents> {
|
|
216
216
|
* A Map of of the loaded guests.
|
217
217
|
*/
|
218
218
|
guests: PortMap = new Map();
|
219
|
-
private cachedCapabilityLists: WeakMap<object,
|
220
|
-
new WeakMap();
|
219
|
+
private cachedCapabilityLists: WeakMap<object, Port[]> = new WeakMap();
|
221
220
|
private runtimeContainer: HTMLElement;
|
222
221
|
private guestOptions: PortOptions;
|
223
222
|
private logger: Console = quietConsole;
|
@@ -239,20 +238,20 @@ export class Host extends Emitter<HostEvents> {
|
|
239
238
|
/**
|
240
239
|
* Return all loaded guests.
|
241
240
|
*/
|
242
|
-
getLoadedGuests():
|
241
|
+
getLoadedGuests<T = unknown>(): Port<T>[];
|
243
242
|
/**
|
244
243
|
* Return loaded guests which satisfy the passed test function.
|
245
244
|
*/
|
246
|
-
getLoadedGuests(filter: GuestFilter):
|
245
|
+
getLoadedGuests<T = unknown>(filter: GuestFilter): Port<T>[];
|
247
246
|
/**
|
248
247
|
* Return loaded guests which expose the provided {@link CapabilitySpec}.
|
249
248
|
*/
|
250
249
|
getLoadedGuests<Apis extends GuestApis>(
|
251
250
|
capabilities: CapabilitySpec<Apis>
|
252
|
-
):
|
251
|
+
): Port<GuestApis>[];
|
253
252
|
getLoadedGuests<Apis extends GuestApis = never>(
|
254
253
|
filterOrCapabilities?: CapabilitySpec<Apis> | GuestFilter
|
255
|
-
):
|
254
|
+
): Port<GuestApis>[] {
|
256
255
|
if (typeof filterOrCapabilities === "object") {
|
257
256
|
return this.getLoadedGuestsWith<Apis>(filterOrCapabilities);
|
258
257
|
}
|
@@ -260,7 +259,7 @@ export class Host extends Emitter<HostEvents> {
|
|
260
259
|
const result = [];
|
261
260
|
for (const guest of this.guests.values()) {
|
262
261
|
if (guest.isReady() && filter(guest)) {
|
263
|
-
result.push(guest);
|
262
|
+
result.push(guest as Port<GuestApis>);
|
264
263
|
}
|
265
264
|
}
|
266
265
|
return result;
|
@@ -345,8 +344,8 @@ export class Host extends Emitter<HostEvents> {
|
|
345
344
|
): Promise<void> {
|
346
345
|
this.runtimeContainer =
|
347
346
|
this.runtimeContainer || this.createRuntimeContainer(window);
|
348
|
-
const failed:
|
349
|
-
const loaded:
|
347
|
+
const failed: Port[] = [];
|
348
|
+
const loaded: Port[] = [];
|
350
349
|
this.loading = true;
|
351
350
|
await Promise.all(
|
352
351
|
Object.entries(extensions).map(async ([id, url]) => {
|
@@ -378,11 +377,11 @@ export class Host extends Emitter<HostEvents> {
|
|
378
377
|
document.body.appendChild(container);
|
379
378
|
return container;
|
380
379
|
}
|
381
|
-
private async loadOneGuest(
|
380
|
+
private async loadOneGuest<T = unknown>(
|
382
381
|
id: string,
|
383
382
|
urlString: string,
|
384
383
|
options: PortOptions = {}
|
385
|
-
): Promise<
|
384
|
+
): Promise<Port<T>> {
|
386
385
|
let guest = this.guests.get(id);
|
387
386
|
if (!guest) {
|
388
387
|
const url = new URL(urlString);
|
package/src/port.ts
CHANGED
@@ -13,13 +13,14 @@ governing permissions and limitations under the License.
|
|
13
13
|
import type {
|
14
14
|
Emits,
|
15
15
|
GuestConnection,
|
16
|
+
GuestConnectionEvents,
|
16
17
|
HostMethodAddress,
|
17
|
-
NamedEvent,
|
18
18
|
RemoteHostApis,
|
19
19
|
GuestApis,
|
20
20
|
CrossRealmObject,
|
21
21
|
Unsubscriber,
|
22
22
|
VirtualApi,
|
23
|
+
UIHostMethods,
|
23
24
|
} from "@adobe/uix-core";
|
24
25
|
import {
|
25
26
|
Emitter,
|
@@ -93,28 +94,6 @@ interface GuestProxyWrapper {
|
|
93
94
|
// #endregion Public Methods (1)
|
94
95
|
}
|
95
96
|
|
96
|
-
/** @public */
|
97
|
-
type PortEvent<
|
98
|
-
GuestApi,
|
99
|
-
Type extends string = string,
|
100
|
-
Detail = Record<string, unknown>
|
101
|
-
> = NamedEvent<
|
102
|
-
Type,
|
103
|
-
Detail &
|
104
|
-
Record<string, unknown> & {
|
105
|
-
guestPort: Port<GuestApi>;
|
106
|
-
}
|
107
|
-
>;
|
108
|
-
|
109
|
-
/** @public */
|
110
|
-
export type PortEvents<
|
111
|
-
GuestApi,
|
112
|
-
HostApi extends Record<string, unknown> = Record<string, unknown>
|
113
|
-
> =
|
114
|
-
| PortEvent<GuestApi, "hostprovide">
|
115
|
-
| PortEvent<GuestApi, "unload">
|
116
|
-
| PortEvent<GuestApi, "beforecallhostmethod", HostMethodAddress<HostApi>>;
|
117
|
-
|
118
97
|
/** @public */
|
119
98
|
export type PortOptions = {
|
120
99
|
/**
|
@@ -152,8 +131,8 @@ const defaultOptions = {
|
|
152
131
|
* something we should review.
|
153
132
|
* @public
|
154
133
|
*/
|
155
|
-
export class Port<GuestApi>
|
156
|
-
extends Emitter<
|
134
|
+
export class Port<GuestApi = unknown>
|
135
|
+
extends Emitter<GuestConnectionEvents>
|
157
136
|
implements GuestConnection
|
158
137
|
{
|
159
138
|
public get apis() {
|
@@ -251,8 +230,18 @@ export class Port<GuestApi>
|
|
251
230
|
* Connect an iframe element which is displaying another page in the extension
|
252
231
|
* with the extension's bootstrap frame, so they can share context and events.
|
253
232
|
*/
|
254
|
-
public attachUI
|
255
|
-
|
233
|
+
public attachUI<T = unknown>(
|
234
|
+
iframe: HTMLIFrameElement
|
235
|
+
): Promise<CrossRealmObject<T>> {
|
236
|
+
return this.attachFrame(iframe, {
|
237
|
+
onIframeResize: (dimensions: { height: number; width: number }) => {
|
238
|
+
this.emit("guestresize", {
|
239
|
+
dimensions,
|
240
|
+
guestPort: this,
|
241
|
+
iframe: iframe,
|
242
|
+
});
|
243
|
+
},
|
244
|
+
} as UIHostMethods);
|
256
245
|
}
|
257
246
|
|
258
247
|
/**
|
@@ -306,9 +295,13 @@ export class Port<GuestApi>
|
|
306
295
|
/**
|
307
296
|
* The host-side equivalent of {@link @adobe/uix-guest#register}. Pass a set
|
308
297
|
* of methods down to the guest as proxies.
|
298
|
+
* Merges at the first level, the API level. Overwrites a deeper levels.
|
309
299
|
*/
|
310
300
|
public provide(apis: RemoteHostApis) {
|
311
|
-
Object.
|
301
|
+
for (const [apiNamespace, methods] of Object.entries(apis)) {
|
302
|
+
this.hostApis[apiNamespace] = this.hostApis[apiNamespace] || {};
|
303
|
+
Object.assign(this.hostApis[apiNamespace], methods);
|
304
|
+
}
|
312
305
|
this.emit("hostprovide", { guestPort: this, apis });
|
313
306
|
}
|
314
307
|
|
@@ -342,9 +335,13 @@ export class Port<GuestApi>
|
|
342
335
|
this.assert(this.isReady(), () => "Attempted to interact before loaded");
|
343
336
|
}
|
344
337
|
|
345
|
-
private attachFrame<T = unknown>(
|
338
|
+
private attachFrame<T = unknown>(
|
339
|
+
iframe: HTMLIFrameElement,
|
340
|
+
addedMethods: object = {}
|
341
|
+
) {
|
346
342
|
// at least this is necessary
|
347
343
|
normalizeIframe(iframe);
|
344
|
+
this.logger.log("attachFrame", iframe);
|
348
345
|
return connectIframe<T>(
|
349
346
|
iframe,
|
350
347
|
{
|
@@ -356,6 +353,7 @@ export class Port<GuestApi>
|
|
356
353
|
getSharedContext: () => this.sharedContext,
|
357
354
|
invokeHostMethod: (address: HostMethodAddress) =>
|
358
355
|
this.invokeHostMethod(address),
|
356
|
+
...addedMethods,
|
359
357
|
}
|
360
358
|
);
|
361
359
|
}
|