@telperion/messenger 7.6.1

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.
Files changed (136) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +600 -0
  3. package/broadcast/default-channel-name.cjs +4 -0
  4. package/broadcast/default-channel-name.d.ts +1 -0
  5. package/broadcast/default-channel-name.js +3 -0
  6. package/broadcast/default-channel-name.js.map +1 -0
  7. package/broadcast/index.cjs +13 -0
  8. package/broadcast/index.d.ts +5 -0
  9. package/broadcast/index.js +7 -0
  10. package/broadcast/index.js.map +1 -0
  11. package/broadcast/message-client.cjs +26 -0
  12. package/broadcast/message-client.d.ts +12 -0
  13. package/broadcast/message-client.js +24 -0
  14. package/broadcast/message-client.js.map +1 -0
  15. package/broadcast/message-host.cjs +28 -0
  16. package/broadcast/message-host.d.ts +9 -0
  17. package/broadcast/message-host.js +26 -0
  18. package/broadcast/message-host.js.map +1 -0
  19. package/broadcast/message-service.cjs +13 -0
  20. package/broadcast/message-service.d.ts +8 -0
  21. package/broadcast/message-service.js +11 -0
  22. package/broadcast/message-service.js.map +1 -0
  23. package/chrome/default-connection-name.cjs +4 -0
  24. package/chrome/default-connection-name.d.ts +1 -0
  25. package/chrome/default-connection-name.js +3 -0
  26. package/chrome/default-connection-name.js.map +1 -0
  27. package/chrome/index.cjs +11 -0
  28. package/chrome/index.d.ts +4 -0
  29. package/chrome/index.js +6 -0
  30. package/chrome/index.js.map +1 -0
  31. package/chrome/message-client.cjs +41 -0
  32. package/chrome/message-client.d.ts +15 -0
  33. package/chrome/message-client.js +39 -0
  34. package/chrome/message-client.js.map +1 -0
  35. package/chrome/message-host.cjs +43 -0
  36. package/chrome/message-host.d.ts +9 -0
  37. package/chrome/message-host.js +41 -0
  38. package/chrome/message-host.js.map +1 -0
  39. package/docs/images/broadcast-communication.svg +87 -0
  40. package/docs/images/chrome-extension-communication.svg +110 -0
  41. package/docs/images/core-architecture.svg +59 -0
  42. package/docs/images/iframe-communication.svg +74 -0
  43. package/docs/images/worker-communication.svg +75 -0
  44. package/iframe/channel-path-splitter.cjs +4 -0
  45. package/iframe/channel-path-splitter.d.ts +1 -0
  46. package/iframe/channel-path-splitter.js +3 -0
  47. package/iframe/channel-path-splitter.js.map +1 -0
  48. package/iframe/default-channel-name.cjs +4 -0
  49. package/iframe/default-channel-name.d.ts +1 -0
  50. package/iframe/default-channel-name.js +3 -0
  51. package/iframe/default-channel-name.js.map +1 -0
  52. package/iframe/iframe.type.cjs +2 -0
  53. package/iframe/iframe.type.d.ts +1 -0
  54. package/iframe/iframe.type.js +3 -0
  55. package/iframe/iframe.type.js.map +1 -0
  56. package/iframe/index.cjs +13 -0
  57. package/iframe/index.d.ts +6 -0
  58. package/iframe/index.js +7 -0
  59. package/iframe/index.js.map +1 -0
  60. package/iframe/message-client.cjs +50 -0
  61. package/iframe/message-client.d.ts +14 -0
  62. package/iframe/message-client.js +48 -0
  63. package/iframe/message-client.js.map +1 -0
  64. package/iframe/message-host.cjs +63 -0
  65. package/iframe/message-host.d.ts +11 -0
  66. package/iframe/message-host.js +61 -0
  67. package/iframe/message-host.js.map +1 -0
  68. package/iframe/message-service.cjs +13 -0
  69. package/iframe/message-service.d.ts +9 -0
  70. package/iframe/message-service.js +11 -0
  71. package/iframe/message-service.js.map +1 -0
  72. package/iframe/source-id-splitter.cjs +4 -0
  73. package/iframe/source-id-splitter.d.ts +1 -0
  74. package/iframe/source-id-splitter.js +3 -0
  75. package/iframe/source-id-splitter.js.map +1 -0
  76. package/iframe/upcoming-message.cjs +2 -0
  77. package/iframe/upcoming-message.d.ts +4 -0
  78. package/iframe/upcoming-message.js +3 -0
  79. package/iframe/upcoming-message.js.map +1 -0
  80. package/index.cjs +13 -0
  81. package/index.d.ts +7 -0
  82. package/index.js +7 -0
  83. package/index.js.map +1 -0
  84. package/listen.decorator.cjs +20 -0
  85. package/listen.decorator.d.ts +6 -0
  86. package/listen.decorator.js +19 -0
  87. package/listen.decorator.js.map +1 -0
  88. package/listener-storage.type.cjs +2 -0
  89. package/listener-storage.type.d.ts +2 -0
  90. package/listener-storage.type.js +3 -0
  91. package/listener-storage.type.js.map +1 -0
  92. package/message-client.cjs +10 -0
  93. package/message-client.d.ts +23 -0
  94. package/message-client.js +8 -0
  95. package/message-client.js.map +1 -0
  96. package/message-host.cjs +58 -0
  97. package/message-host.d.ts +26 -0
  98. package/message-host.js +56 -0
  99. package/message-host.js.map +1 -0
  100. package/message-response.type.cjs +2 -0
  101. package/message-response.type.d.ts +16 -0
  102. package/message-response.type.js +3 -0
  103. package/message-response.type.js.map +1 -0
  104. package/message.interface.cjs +2 -0
  105. package/message.interface.d.ts +6 -0
  106. package/message.interface.js +3 -0
  107. package/message.interface.js.map +1 -0
  108. package/package.json +158 -0
  109. package/request.decorator.cjs +20 -0
  110. package/request.decorator.d.ts +1 -0
  111. package/request.decorator.js +19 -0
  112. package/request.decorator.js.map +1 -0
  113. package/selectors.cjs +10 -0
  114. package/selectors.d.ts +7 -0
  115. package/selectors.js +9 -0
  116. package/selectors.js.map +1 -0
  117. package/worker/index.cjs +13 -0
  118. package/worker/index.d.ts +5 -0
  119. package/worker/index.js +7 -0
  120. package/worker/index.js.map +1 -0
  121. package/worker/initializer.cjs +39 -0
  122. package/worker/initializer.d.ts +12 -0
  123. package/worker/initializer.js +38 -0
  124. package/worker/initializer.js.map +1 -0
  125. package/worker/message-client.cjs +158 -0
  126. package/worker/message-client.d.ts +131 -0
  127. package/worker/message-client.js +156 -0
  128. package/worker/message-client.js.map +1 -0
  129. package/worker/message-host.cjs +165 -0
  130. package/worker/message-host.d.ts +131 -0
  131. package/worker/message-host.js +163 -0
  132. package/worker/message-host.js.map +1 -0
  133. package/worker/message-service.cjs +123 -0
  134. package/worker/message-service.d.ts +117 -0
  135. package/worker/message-service.js +121 -0
  136. package/worker/message-service.js.map +1 -0
package/LICENSE ADDED
@@ -0,0 +1,21 @@
1
+ MIT License
2
+
3
+ Copyright (c) 2019 Thalesrc
4
+
5
+ Permission is hereby granted, free of charge, to any person obtaining a copy
6
+ of this software and associated documentation files (the "Software"), to deal
7
+ in the Software without restriction, including without limitation the rights
8
+ to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9
+ copies of the Software, and to permit persons to whom the Software is
10
+ furnished to do so, subject to the following conditions:
11
+
12
+ The above copyright notice and this permission notice shall be included in all
13
+ copies or substantial portions of the Software.
14
+
15
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16
+ IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17
+ FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18
+ AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19
+ LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20
+ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21
+ SOFTWARE.
package/README.md ADDED
@@ -0,0 +1,600 @@
1
+ # @telperion/messenger
2
+
3
+ [![npm](https://img.shields.io/npm/v/@telperion/messenger.svg)](https://www.npmjs.com/package/@telperion/messenger)
4
+ [![npm](https://img.shields.io/npm/dw/@telperion/messenger.svg)](https://www.npmjs.com/package/@telperion/messenger)
5
+ [![codecov](https://codecov.io/gh/telperiontech/telperion/graph/badge.svg?token=dz46LY3onk&flag=messenger)](https://codecov.io/gh/telperiontech/telperion?flag=messenger)
6
+ [![TypeScript](https://badges.frapsoft.com/typescript/version/typescript-next.svg?v=101)](https://www.typescriptlang.org/)
7
+ [![License: MIT](https://img.shields.io/badge/License-MIT-yellow.svg)](https://opensource.org/licenses/MIT)
8
+
9
+ Javascript messaging library for cross-context communication
10
+
11
+ **Part of the [Telperion](https://github.com/telperiontech/telperion) monorepo**
12
+
13
+ ## Installation
14
+
15
+ ```bash
16
+ npm install @telperion/messenger
17
+ # or
18
+ yarn add @telperion/messenger
19
+ # or
20
+ pnpm add @telperion/messenger
21
+ ```
22
+
23
+ ## Overview
24
+
25
+ Messenger provides a unified, decorator-based API for cross-context messaging in JavaScript applications. Built on RxJS, it supports:
26
+
27
+ - 🖼️ **Iframe Communication** - Parent-child window messaging
28
+ - 🧩 **Chrome Extensions** - Background scripts, content scripts, and popups
29
+ - 👷 **Web Workers** - Main thread and worker communication
30
+ - 📡 **Broadcast Channel** - Tab-to-tab messaging
31
+
32
+ ## Core Concepts
33
+
34
+ ### MessageClient & MessageHost
35
+
36
+ The library uses two main abstractions:
37
+
38
+ - **`MessageClient`**: Sends requests and receives responses (uses `@Request` decorator)
39
+ - **`MessageHost`**: Listens for requests and sends responses (uses `@Listen` decorator)
40
+ - **`MessageService`**: Combines both client and host (bidirectional communication)
41
+
42
+ ![Core Architecture](docs/images/core-architecture.svg)
43
+
44
+ ### Decorators
45
+
46
+ - **`@Request(path)`**: Marks a method as a message sender
47
+ - **`@Listen(path)`**: Marks a method as a message listener
48
+
49
+ ## Usage
50
+
51
+ ### Iframe Communication
52
+
53
+ Send and receive messages between iframes and parent windows.
54
+
55
+ ![Iframe Communication](docs/images/iframe-communication.svg)
56
+
57
+ #### Client-Only (Iframe)
58
+
59
+ ```typescript
60
+ import { IframeMessageClient, Request } from '@telperion/messenger/iframe';
61
+ import { Observable } from 'rxjs';
62
+
63
+ class IframeClient extends IframeMessageClient {
64
+ @Request('greeting')
65
+ sayHello(name: string): Observable<string> {
66
+ return null!; // Implementation handled by decorator
67
+ }
68
+ }
69
+
70
+ // Default: sends messages to parent window
71
+ const client = new IframeClient();
72
+
73
+ // Optional: specify channel name
74
+ const clientWithChannel = new IframeClient('my-channel');
75
+
76
+ // Optional: target specific iframe (from parent window)
77
+ const iframe = document.querySelector('iframe');
78
+ const clientToIframe = new IframeClient('my-channel', iframe);
79
+
80
+ // Optional: use a function to get iframe dynamically
81
+ const clientWithDynamicIframe = new IframeClient('my-channel', () =>
82
+ document.querySelector('iframe#dynamic')
83
+ );
84
+
85
+ client.sayHello('John').subscribe(response => {
86
+ console.log(response); // 'Hello John!'
87
+ });
88
+ ```
89
+
90
+ #### Host-Only (Parent Window)
91
+
92
+ ```typescript
93
+ import { IframeMessageHost, Listen, UpcomingMessage } from '@telperion/messenger/iframe';
94
+ import { of } from 'rxjs';
95
+
96
+ class ParentHost extends IframeMessageHost {
97
+ @Listen('greeting')
98
+ handleGreeting({ data }: UpcomingMessage<string>): Observable<string> {
99
+ return of(`Hello ${data}!`);
100
+ }
101
+ }
102
+
103
+ // Default: listens to all iframes
104
+ const host = new ParentHost();
105
+
106
+ // Optional: specify channel name
107
+ const hostWithChannel = new ParentHost('my-channel');
108
+
109
+ // Optional: listen only to specific iframe
110
+ const iframe = document.querySelector('iframe');
111
+ const hostForSpecificIframe = new ParentHost('my-channel', iframe);
112
+
113
+ // Optional: use a function to get iframe dynamically
114
+ const hostWithDynamicIframe = new ParentHost('my-channel', () =>
115
+ document.querySelector('iframe#dynamic')
116
+ );
117
+ ```
118
+
119
+ #### Bidirectional Communication (MessageService)
120
+
121
+ ```typescript
122
+ import { IframeMessageService, Request, Listen } from '@telperion/messenger/iframe';
123
+ import { of, Observable } from 'rxjs';
124
+
125
+ class IframeBidirectional extends IframeMessageService {
126
+ // Send messages
127
+ @Request('getData')
128
+ requestData(): Observable<any> {
129
+ return null!;
130
+ }
131
+
132
+ // Receive messages
133
+ @Listen('update')
134
+ handleUpdate(data: any): Observable<string> {
135
+ console.log('Received update:', data);
136
+ return of('Update processed');
137
+ }
138
+ }
139
+
140
+ // Default: communicates with parent window
141
+ const service = new IframeBidirectional();
142
+
143
+ // Optional: specify channel name and target frame
144
+ const iframe = document.querySelector('iframe');
145
+ const serviceWithTarget = new IframeBidirectional('my-channel', iframe);
146
+
147
+ // From within iframe (communicates with parent)
148
+ const iframeService = new IframeBidirectional('my-channel');
149
+ ```
150
+
151
+ ---
152
+
153
+ ### Chrome Extensions
154
+
155
+ Communicate across extension contexts (background, content scripts, popups).
156
+
157
+ ![Chrome Extension Communication](docs/images/chrome-extension-communication.svg)
158
+
159
+ #### Content Script
160
+
161
+ ```typescript
162
+ import { ChromeMessageClient, Request } from '@telperion/messenger/chrome';
163
+ import { Observable } from 'rxjs';
164
+
165
+ class ContentScript extends ChromeMessageClient {
166
+ @Request('fetchData')
167
+ getData(query: string): Observable<any> {
168
+ return null!;
169
+ }
170
+
171
+ @Request('saveSettings')
172
+ saveSettings(settings: object): Observable<boolean> {
173
+ return null!;
174
+ }
175
+ }
176
+
177
+ const contentScript = new ContentScript();
178
+
179
+ contentScript.getData('user').subscribe(data => {
180
+ console.log('Received:', data);
181
+ });
182
+ ```
183
+
184
+ #### Background Script
185
+
186
+ ```typescript
187
+ import { ChromeMessageHost, Listen } from '@telperion/messenger/chrome';
188
+ import { of, Observable } from 'rxjs';
189
+
190
+ class BackgroundScript extends ChromeMessageHost {
191
+ @Listen('fetchData')
192
+ handleFetchData(query: string): Observable<any> {
193
+ // Fetch from API or storage
194
+ return of({ name: 'John', age: 30 });
195
+ }
196
+
197
+ @Listen('saveSettings')
198
+ handleSaveSettings(settings: object): Observable<boolean> {
199
+ // Save to chrome.storage
200
+ return of(true);
201
+ }
202
+ }
203
+
204
+ const background = new BackgroundScript();
205
+ ```
206
+
207
+ ---
208
+
209
+ ### Web Workers
210
+
211
+ Communicate between main thread and web workers.
212
+
213
+ ![Web Worker Communication](docs/images/worker-communication.svg)
214
+
215
+ #### Main Thread
216
+
217
+ ```typescript
218
+ import { WorkerMessageService, Request, Listen } from '@telperion/messenger/worker';
219
+ import { of, Observable } from 'rxjs';
220
+
221
+ class MainThread extends WorkerMessageService {
222
+ @Request('processData')
223
+ sendDataToWorker(data: number[]): Observable<number> {
224
+ return null!;
225
+ }
226
+
227
+ @Listen('progress')
228
+ handleProgress(percent: number): Observable<void> {
229
+ console.log(`Progress: ${percent}%`);
230
+ return of(void 0);
231
+ }
232
+ }
233
+
234
+ // Must provide Worker instance in main thread
235
+ const worker = new Worker('./worker.js');
236
+ const mainService = new MainThread(worker);
237
+
238
+ mainService.sendDataToWorker([1, 2, 3, 4, 5]).subscribe(result => {
239
+ console.log('Worker result:', result);
240
+ });
241
+ ```
242
+
243
+ ##### Flexible Worker Initialization
244
+
245
+ The worker parameter supports multiple initialization patterns for different use cases:
246
+
247
+ ```typescript
248
+ // Direct Worker instance
249
+ const service1 = new MainThread(new Worker('./worker.js'));
250
+
251
+ // Promise that resolves to a Worker (for async initialization)
252
+ const workerPromise = import('./worker.js').then(m => new Worker(m.default));
253
+ const service2 = new MainThread(workerPromise);
254
+
255
+ // Function that returns a Worker (for lazy initialization)
256
+ const service3 = new MainThread(() => new Worker('./worker.js'));
257
+
258
+ // Function that returns a Promise<Worker> (for async lazy initialization)
259
+ const service4 = new MainThread(async () => {
260
+ const module = await import('./worker.js');
261
+ return new Worker(module.default);
262
+ });
263
+ ```
264
+
265
+ ##### Dynamic Worker Management with `initialize()`
266
+
267
+ The `initialize()` method allows you to switch workers at runtime or re-establish connections:
268
+
269
+ ```typescript
270
+ class MainThread extends WorkerMessageService {
271
+ // ... decorators and methods
272
+ }
273
+
274
+ // Start without a worker
275
+ const service = new MainThread();
276
+
277
+ // Later, connect to a worker dynamically
278
+ service.initialize(new Worker('./worker.js'));
279
+
280
+ // Switch to a different worker
281
+ service.initialize(new Worker('./different-worker.js'));
282
+
283
+ // Re-establish connection after worker error
284
+ const worker = new Worker('./worker.js');
285
+ worker.onerror = (error) => {
286
+ console.error('Worker error, reinitializing...', error);
287
+ service.initialize(new Worker('./worker.js'));
288
+ };
289
+ service.initialize(worker);
290
+
291
+ // Conditional worker initialization
292
+ function getWorker() {
293
+ return navigator.hardwareConcurrency > 2
294
+ ? new Worker('./heavy-worker.js')
295
+ : new Worker('./light-worker.js');
296
+ }
297
+ service.initialize(getWorker);
298
+ ```
299
+
300
+ #### Worker Thread
301
+
302
+ ```typescript
303
+ import { WorkerMessageService, Request, Listen } from '@telperion/messenger/worker';
304
+ import { of, Observable } from 'rxjs';
305
+
306
+ class WorkerThread extends WorkerMessageService {
307
+ @Listen('processData')
308
+ handleProcessData(data: number[]): Observable<number> {
309
+ // Heavy computation
310
+ const result = data.reduce((sum, n) => sum + n, 0);
311
+ return of(result);
312
+ }
313
+
314
+ @Request('progress')
315
+ reportProgress(percent: number): Observable<void> {
316
+ return null!;
317
+ }
318
+ }
319
+
320
+ // Inside worker: no argument needed (uses self)
321
+ const workerService = new WorkerThread();
322
+ ```
323
+
324
+ ---
325
+
326
+ ### Broadcast Channel
327
+
328
+ Communicate between different tabs/windows of the same origin.
329
+
330
+ ![Broadcast Channel Communication](docs/images/broadcast-communication.svg)
331
+
332
+ #### Tab 1
333
+
334
+ ```typescript
335
+ import { BroadcastMessageService, Request, Listen } from '@telperion/messenger/broadcast';
336
+ import { of, Observable } from 'rxjs';
337
+
338
+ class Tab1 extends BroadcastMessageService {
339
+ @Request('sync')
340
+ requestSync(data: any): Observable<string> {
341
+ return null!;
342
+ }
343
+
344
+ @Listen('notification')
345
+ handleNotification(message: string): Observable<void> {
346
+ console.log('Notification:', message);
347
+ return of(void 0);
348
+ }
349
+ }
350
+
351
+ const tab1 = new Tab1('my-app-channel');
352
+
353
+ tab1.requestSync({ user: 'John' }).subscribe(response => {
354
+ console.log(response); // 'Sync completed'
355
+ });
356
+ ```
357
+
358
+ #### Tab 2
359
+
360
+ ```typescript
361
+ import { BroadcastMessageService, Request, Listen } from '@telperion/messenger/broadcast';
362
+ import { of, Observable } from 'rxjs';
363
+
364
+ class Tab2 extends BroadcastMessageService {
365
+ @Listen('sync')
366
+ handleSync(data: any): Observable<string> {
367
+ console.log('Syncing data:', data);
368
+ return of('Sync completed');
369
+ }
370
+
371
+ @Request('notification')
372
+ sendNotification(message: string): Observable<void> {
373
+ return null!;
374
+ }
375
+ }
376
+
377
+ const tab2 = new Tab2('my-app-channel');
378
+ ```
379
+
380
+ ---
381
+
382
+ ## Advanced Features
383
+
384
+ ### Streaming Responses
385
+
386
+ Return multiple values over time using RxJS operators:
387
+
388
+ ```typescript
389
+ import { Listen } from '@telperion/messenger/iframe';
390
+ import { interval } from 'rxjs';
391
+ import { map, take } from 'rxjs/operators';
392
+
393
+ class StreamingHost extends IframeMessageHost {
394
+ @Listen('countdown')
395
+ handleCountdown(start: number): Observable<number> {
396
+ return interval(1000).pipe(
397
+ map(i => start - i),
398
+ take(start + 1)
399
+ );
400
+ }
401
+ }
402
+ ```
403
+
404
+ The client receives each value as it's emitted:
405
+
406
+ ```typescript
407
+ client.countdown(5).subscribe(
408
+ value => console.log(value), // 5, 4, 3, 2, 1, 0
409
+ error => console.error(error),
410
+ () => console.log('Complete!')
411
+ );
412
+ ```
413
+
414
+ ### Error Handling
415
+
416
+ ```typescript
417
+ import { Listen } from '@telperion/messenger/iframe';
418
+ import { throwError } from 'rxjs';
419
+
420
+ class ErrorHost extends IframeMessageHost {
421
+ @Listen('riskyOperation')
422
+ handleRiskyOperation(data: any): Observable<any> {
423
+ if (!data.valid) {
424
+ return throwError(() => new Error('Invalid data'));
425
+ }
426
+ return of({ success: true });
427
+ }
428
+ }
429
+ ```
430
+
431
+ ### Constructor Parameters
432
+
433
+ #### Iframe
434
+
435
+ **`IframeMessageClient` / `IframeMessageHost` / `IframeMessageService`**
436
+
437
+ ```typescript
438
+ constructor(channelName?: string, targetFrame?: HTMLIFrameElement | (() => HTMLIFrameElement))
439
+ ```
440
+
441
+ - **`channelName`** (optional): Channel identifier for namespacing messages. Default: `'messenger-iframe-message'`
442
+ - **`targetFrame`** (optional): Specific iframe to communicate with. Can be:
443
+ - `HTMLIFrameElement`: Direct reference to iframe element
444
+ - `() => HTMLIFrameElement`: Function returning iframe (useful for dynamic iframes)
445
+ - Omit to communicate with parent window (from iframe) or all iframes (from parent)
446
+
447
+ **Examples:**
448
+ ```typescript
449
+ // From iframe: communicate with parent
450
+ const client = new IframeMessageClient();
451
+
452
+ // From parent: communicate with specific iframe
453
+ const iframe = document.querySelector('iframe');
454
+ const host = new IframeMessageHost('my-channel', iframe);
455
+
456
+ // Dynamic iframe reference
457
+ const service = new IframeMessageService('my-channel', () =>
458
+ document.querySelector('iframe[data-active="true"]')
459
+ );
460
+ ```
461
+
462
+ #### Worker
463
+
464
+ **`WorkerMessageClient` / `WorkerMessageHost` / `WorkerMessageService`**
465
+
466
+ ```typescript
467
+ constructor(worker?: Worker)
468
+ ```
469
+
470
+ - **`worker`** (optional): Worker instance for main thread communication
471
+ - **In main thread**: Must provide `Worker` instance
472
+ - **In worker thread**: Omit parameter (uses `self` automatically)
473
+
474
+ **Examples:**
475
+ ```typescript
476
+ // Main thread: must provide worker
477
+ const worker = new Worker('./worker.js');
478
+ const service = new WorkerMessageService(worker);
479
+
480
+ // Inside worker: no parameter needed
481
+ const service = new WorkerMessageService();
482
+ ```
483
+
484
+ #### Broadcast
485
+
486
+ **`BroadcastMessageClient` / `BroadcastMessageHost` / `BroadcastMessageService`**
487
+
488
+ ```typescript
489
+ constructor(channelName?: string)
490
+ ```
491
+
492
+ - **`channelName`** (optional): Broadcast channel name. Default: `'messenger-broadcast-message'`
493
+
494
+ **Example:**
495
+ ```typescript
496
+ const service = new BroadcastMessageService('app-sync-channel');
497
+ ```
498
+
499
+ #### Chrome
500
+
501
+ **`ChromeMessageClient` / `ChromeMessageHost`**
502
+
503
+ ```typescript
504
+ constructor(connectionName?: string)
505
+ ```
506
+
507
+ - **`connectionName`** (optional): Connection identifier. Default: `'messenger-chrome-message'`
508
+
509
+ **Example:**
510
+ ```typescript
511
+ const client = new ChromeMessageClient('extension-port');
512
+ ```
513
+
514
+ ## API Reference
515
+
516
+ ### Classes
517
+
518
+ - **`MessageClient`** - Base class for message senders
519
+ - **`MessageHost`** - Base class for message receivers
520
+ - **`IframeMessageClient`** - Iframe client implementation
521
+ - **`IframeMessageHost`** - Iframe host implementation
522
+ - **`IframeMessageService`** - Bidirectional iframe communication
523
+ - **`ChromeMessageClient`** - Chrome extension client
524
+ - **`ChromeMessageHost`** - Chrome extension host
525
+ - **`WorkerMessageService`** - Bidirectional worker communication
526
+ - **`BroadcastMessageClient`** - Broadcast channel client
527
+ - **`BroadcastMessageHost`** - Broadcast channel host
528
+ - **`BroadcastMessageService`** - Bidirectional broadcast communication
529
+
530
+ ### Decorators
531
+
532
+ - **`@Request(path: string)`** - Decorator for sending messages
533
+ - **`@Listen(path: string)`** - Decorator for receiving messages
534
+
535
+ ### Types
536
+
537
+ - **`Message`** - Message payload structure
538
+ - **`MessageResponse`** - Response payload structure
539
+ - **`UpcomingMessage<T>`** - Incoming message with sender info (iframe)
540
+
541
+ ## Requirements
542
+
543
+ - RxJS 7.x or higher
544
+ - TypeScript 4.x or higher (for decorator support)
545
+ - Modern browser with ES2015+ support
546
+
547
+ ## Development
548
+
549
+ ### Testing
550
+
551
+ Messenger uses a modular testing structure with separate configurations for each submodule:
552
+
553
+ ```bash
554
+ # Run all tests (all submodules in parallel)
555
+ pnpm nx run messenger:test
556
+
557
+ # Run specific submodule tests
558
+ pnpm nx run messenger:test/core # Core functionality (Node.js)
559
+ pnpm nx run messenger:test/worker # Web Worker tests (Browser)
560
+ pnpm nx run messenger:test/chrome # Chrome extension tests (Browser)
561
+ pnpm nx run messenger:test/broadcast # Broadcast Channel tests (Browser)
562
+ pnpm nx run messenger:test/iframe # Iframe communication tests (Browser)
563
+
564
+ # Run all tests with coverage and merge reports
565
+ pnpm nx run messenger:test/coverage
566
+
567
+ # Debug browser tests (visible browser)
568
+ pnpm nx run messenger:test/headed
569
+ ```
570
+
571
+ For detailed testing documentation, see:
572
+ - [TESTING-GUIDE.md](./docs/TESTING-GUIDE.md) - Complete testing guide
573
+ - [TEST-COMMANDS.md](./docs/TEST-COMMANDS.md) - Quick command reference
574
+ - [TESTING-SETUP-SUMMARY.md](./docs/TESTING-SETUP-SUMMARY.md) - Setup overview
575
+
576
+ ### Building
577
+
578
+ ```bash
579
+ # Build the library
580
+ pnpm nx build messenger
581
+
582
+ # Build and watch for changes
583
+ pnpm nx build messenger --watch
584
+ ```
585
+
586
+ ### Contributing
587
+
588
+ When adding new submodule support:
589
+
590
+ 1. Create submodule directory in `src/`
591
+ 2. Add test files alongside code: `*.spec.ts`
592
+ 3. Create submodule's `vitest.config.ts` in the submodule directory
593
+ 4. Add test target in `project.json`
594
+ 5. Update main `test` target to include new submodule
595
+
596
+ See [TESTING-GUIDE.md](./docs/TESTING-GUIDE.md) for details on adding new submodules.
597
+
598
+ ## License
599
+
600
+ MIT © [Telperion](https://github.com/telperiontech)
@@ -0,0 +1,4 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.DEFAULT_CHANNEL_NAME = void 0;
4
+ exports.DEFAULT_CHANNEL_NAME = 'MESSENGER_DEFAULT_CHANNEL';
@@ -0,0 +1 @@
1
+ export declare const DEFAULT_CHANNEL_NAME = "MESSENGER_DEFAULT_CHANNEL";
@@ -0,0 +1,3 @@
1
+ export const DEFAULT_CHANNEL_NAME = 'MESSENGER_DEFAULT_CHANNEL';
2
+
3
+ //# sourceMappingURL=default-channel-name.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../../../../../libs/messenger/src/broadcast/default-channel-name.ts"],"names":[],"mappings":"AAAA,MAAM,CAAC,MAAM,oBAAoB,GAAG,2BAA2B,CAAC","file":"default-channel-name.js","sourcesContent":["export const DEFAULT_CHANNEL_NAME = 'MESSENGER_DEFAULT_CHANNEL';\n"]}
@@ -0,0 +1,13 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.Listen = exports.Request = exports.BroadcastMessageService = exports.BroadcastMessageHost = exports.BroadcastMessageClient = void 0;
4
+ var message_client_1 = require("./message-client");
5
+ Object.defineProperty(exports, "BroadcastMessageClient", { enumerable: true, get: function () { return message_client_1.BroadcastMessageClient; } });
6
+ var message_host_1 = require("./message-host");
7
+ Object.defineProperty(exports, "BroadcastMessageHost", { enumerable: true, get: function () { return message_host_1.BroadcastMessageHost; } });
8
+ var message_service_1 = require("./message-service");
9
+ Object.defineProperty(exports, "BroadcastMessageService", { enumerable: true, get: function () { return message_service_1.BroadcastMessageService; } });
10
+ var request_decorator_1 = require("../request.decorator");
11
+ Object.defineProperty(exports, "Request", { enumerable: true, get: function () { return request_decorator_1.Request; } });
12
+ var listen_decorator_1 = require("../listen.decorator");
13
+ Object.defineProperty(exports, "Listen", { enumerable: true, get: function () { return listen_decorator_1.Listen; } });
@@ -0,0 +1,5 @@
1
+ export { BroadcastMessageClient } from './message-client';
2
+ export { BroadcastMessageHost } from './message-host';
3
+ export { BroadcastMessageService } from './message-service';
4
+ export { Request } from '../request.decorator';
5
+ export { Listen } from '../listen.decorator';
@@ -0,0 +1,7 @@
1
+ export { BroadcastMessageClient } from './message-client';
2
+ export { BroadcastMessageHost } from './message-host';
3
+ export { BroadcastMessageService } from './message-service';
4
+ export { Request } from '../request.decorator';
5
+ export { Listen } from '../listen.decorator';
6
+
7
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../../../../../libs/messenger/src/broadcast/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,sBAAsB,EAAE,MAAM,kBAAkB,CAAC;AAC1D,OAAO,EAAE,oBAAoB,EAAE,MAAM,gBAAgB,CAAC;AACtD,OAAO,EAAE,uBAAuB,EAAE,MAAM,mBAAmB,CAAC;AAC5D,OAAO,EAAE,OAAO,EAAE,MAAM,sBAAsB,CAAC;AAC/C,OAAO,EAAE,MAAM,EAAE,MAAM,qBAAqB,CAAC","file":"index.js","sourcesContent":["export { BroadcastMessageClient } from './message-client';\nexport { BroadcastMessageHost } from './message-host';\nexport { BroadcastMessageService } from './message-service';\nexport { Request } from '../request.decorator';\nexport { Listen } from '../listen.decorator';\n"]}
@@ -0,0 +1,26 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.BroadcastMessageClient = void 0;
4
+ const rxjs_1 = require("rxjs");
5
+ const message_client_1 = require("../message-client");
6
+ const selectors_1 = require("../selectors");
7
+ const default_channel_name_1 = require("./default-channel-name");
8
+ class BroadcastMessageClient extends message_client_1.MessageClient {
9
+ [selectors_1.RESPONSES$] = new rxjs_1.Subject();
10
+ #channel;
11
+ constructor(channelName = default_channel_name_1.DEFAULT_CHANNEL_NAME) {
12
+ super();
13
+ this.#channel = new BroadcastChannel(channelName);
14
+ this.#channel.addEventListener('message', this.#handler);
15
+ }
16
+ [selectors_1.SEND](message) {
17
+ this.#channel.postMessage(message);
18
+ }
19
+ #handler = (event) => {
20
+ this[selectors_1.RESPONSES$].next(event.data);
21
+ };
22
+ [selectors_1.GET_NEW_ID]() {
23
+ return crypto.randomUUID();
24
+ }
25
+ }
26
+ exports.BroadcastMessageClient = BroadcastMessageClient;