diff --git a/CHANGELOG.md b/CHANGELOG.md index e414e7982..8751ea4c4 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -23,7 +23,7 @@ The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/). * Added utility functions `isStandardContextType(contextType: string)`, `isStandardIntent(intent: string)`,`getPossibleContextsForIntent(intent: StandardIntent)`. ([#1139](https://github.com/finos/FDC3/pull/1139)) * Added support for event listening outside of intent or context listnener. Added new function `addEventListener`, type `EventHandler`, enum `FDC3EventType` and interfaces `FDC3Event` and `FDC3ChannelChangedEvent`. ([#1207](https://github.com/finos/FDC3/pull/1207)) * Added new `CreateOrUpdateProfile` intent. ([#1359](https://github.com/finos/FDC3/pull/1359)) - +* Added clarification regarding expected behavior upon repeated calls to `addContextListener` on same or overlapping types (allowed) and `addIntentListener` on same intent (rejected; new error type added). ([#1394](https://github.com/finos/FDC3/pull/1394)) ### Changed diff --git a/docs/api/ref/Channel.md b/docs/api/ref/Channel.md index 4f949f09d..454703ee6 100644 --- a/docs/api/ref/Channel.md +++ b/docs/api/ref/Channel.md @@ -174,6 +174,8 @@ If, when this function is called, the channel already contains context that woul Optional metadata about each context message received, including the app that originated the message, SHOULD be provided by the desktop agent implementation. +Adding multiple context listeners on the same or overlapping types (i.e. specific `contextType` and `null` type) MUST be allowed, and MUST trigger all ContextHandlers when a relevant context type is broadcast on the current channel. + **Examples:** Add a listener for any context that is broadcast on the channel: diff --git a/docs/api/ref/DesktopAgent.md b/docs/api/ref/DesktopAgent.md index 9525418b7..c639b5c37 100644 --- a/docs/api/ref/DesktopAgent.md +++ b/docs/api/ref/DesktopAgent.md @@ -132,6 +132,8 @@ Context may also be received via this listener if the application was launched v Optional metadata about each context message received, including the app that originated the message, SHOULD be provided by the Desktop Agent implementation. +Adding multiple context listeners on the same or overlapping types (i.e. specific `contextType` and `null` type) MUST be allowed, and MUST trigger all ContextHandlers when a relevant context type is broadcast on the current user channel. Please note, that this behavior differs from [`fdc3.addIntentListener`](#addintentlistener) call; refer to the relevant documentation for more details. + **Examples:** @@ -264,6 +266,8 @@ The [`PrivateChannel`](PrivateChannel) type is provided to support synchronizati Optional metadata about each intent & context message received, including the app that originated the message, SHOULD be provided by the desktop agent implementation. + Adding multiple intent listeners on the same type MUST be rejected with the [`ResolveError.IntentListenerConflict`](Errors#resolveerror), unless the previous listener was removed first though [`listener.unsubscribe`](Types#unsubscribe) + **Examples:** @@ -339,6 +343,7 @@ var listener = await _desktopAgent.AddIntentListener("StartChat", (con - [`Listener`](Types#listener) - [`Context`](Types#context) - [`IntentHandler`](Types#intenthandler) +- [`ResolveError`](Errors#resolveerror) ### `broadcast` @@ -527,7 +532,7 @@ Task> FindInstances(IAppIdentifier app); Find all the available instances for a particular application. -If the application is not known to the agent, the returned promise should be rejected with the `ResolverError.NoAppsFound` error message. However, if the application is known but there are no instances of the specified app the returned promise should resolve to an empty array. +If the application is not known to the agent, the returned promise should be rejected with the `ResolveError.NoAppsFound` error message. However, if the application is known but there are no instances of the specified app the returned promise should resolve to an empty array. If the request fails for another reason, the promise MUST be rejected with an `Error` Object with a `message` chosen from the [`ResolveError`](Errors#resolveerror) enumeration, or (if connected to a Desktop Agent Bridge) the [`BridgingError`](Errors#bridgingerror) enumeration. diff --git a/docs/api/spec.md b/docs/api/spec.md index b15eec8ce..b597fabfb 100644 --- a/docs/api/spec.md +++ b/docs/api/spec.md @@ -535,6 +535,8 @@ When an instance of an application is launched, it is expected to add an [`Inten Intent handlers SHOULD be registered via [`fdc3.addIntentListener`](ref/DesktopAgent#addintentlistener) within 15 seconds of the application launch (the minimum timeout Desktop Agents are required to provide) in order to be widely compatible with Desktop Agent implementations. Individual Desktop Agent implementations MAY support longer timeouts or configuration to control or extend timeouts. +A single handler can be added for each specific intent. If the application attempts to call [`fdc3.addIntentListener`](ref/DesktopAgent#addintentlistener) passing the same `intent` a second time, before unsubscribing to the previously added listener, the Desktop Agent MUST reject it with an `Error` Object with the message given by [`ResolveError.IntentListenerConflict`](ref/Errors#resolveerror). + ### Originating App Metadata Optional metadata about each intent & context message received, including the app that originated the message, SHOULD be provided by the desktop agent implementation to registered intent handlers. As this metadata is optional, apps making use of it MUST handle cases where it is not provided. diff --git a/schemas/api/api.schema.json b/schemas/api/api.schema.json index 7f1fcf0ad..ac660dc98 100644 --- a/schemas/api/api.schema.json +++ b/schemas/api/api.schema.json @@ -313,7 +313,7 @@ "type": "string" }, "ResolveError": { - "description": "Constants representing the errors that can be encountered when calling the `findIntent`, `findIntentsByContext`, `raiseIntent` or `raiseIntentForContext` methods on the DesktopAgent (`fdc3`).", + "description": "Constants representing the errors that can be encountered when calling the `addIntentListener`, `findIntent`, `findIntentsByContext`, `raiseIntent` or `raiseIntentForContext` methods on the DesktopAgent (`fdc3`).", "title": "ResolveError", "enum": [ "DesktopAgentNotFound", @@ -324,7 +324,8 @@ "ResolverUnavailable", "TargetAppUnavailable", "TargetInstanceUnavailable", - "UserCancelledResolution" + "UserCancelledResolution", + "IntentListenerConflict" ], "type": "string" }, diff --git a/src/api/Channel.ts b/src/api/Channel.ts index 9eca774b5..8a964d538 100644 --- a/src/api/Channel.ts +++ b/src/api/Channel.ts @@ -70,6 +70,9 @@ export interface Channel { * If, when this function is called, the channel already contains context that would be passed to the listener it is NOT called or passed this context automatically (this behavior differs from that of the [`fdc3.addContextListener`](DesktopAgent#addcontextlistener) function). Apps wishing to access to the current context of the channel should instead call the `getCurrentContext(contextType)` function. * * Optional metadata about each context message received, including the app that originated the message, SHOULD be provided by the desktop agent implementation. + * + * Adding multiple context listeners on the same or overlapping types (i.e. named type and null type) MUST be allowed, and MUST trigger all context handlers when a relevant context type is broadcast on the current channel. + * */ addContextListener(contextType: string | null, handler: ContextHandler): Promise; diff --git a/src/api/DesktopAgent.ts b/src/api/DesktopAgent.ts index 3691ae3f6..86c0efc6f 100644 --- a/src/api/DesktopAgent.ts +++ b/src/api/DesktopAgent.ts @@ -305,6 +305,8 @@ export interface DesktopAgent { * The `PrivateChannel` type is provided to support synchronization of data transmitted over returned channels, by allowing both parties to listen for events denoting subscription and unsubscription from the returned channel. `PrivateChannels` are only retrievable via raising an intent. * * Optional metadata about the raised intent, including the app that originated the message, SHOULD be provided by the desktop agent implementation. + * + * Adding multiple intent listeners on the same type MUST be rejected with the `ResolveError.IntentListenerConflict`, unless the previous listener was removed first though `listener.unsubscribe` * * ```javascript * //Handle a raised intent @@ -359,6 +361,8 @@ export interface DesktopAgent { * Context may also be received via this listener if the application was launched via a call to `fdc3.open`, where context was passed as an argument. In order to receive this, applications SHOULD add their context listener as quickly as possible after launch, or an error MAY be returned to the caller and the context may not be delivered. The exact timeout used is set by the Desktop Agent implementation, but MUST be at least 15 seconds. * * Optional metadata about the context message, including the app that originated the message, SHOULD be provided by the desktop agent implementation. + * + * Adding multiple context listeners on the same or overlapping types (i.e. named type and null type) MUST be allowed, and MUST trigger all contextHandlers when a relevant context type is broadcast on the current user channel. Please note, that this behavior differes from `fdc3.addIntentListener`API call; refer to the relevant documentation for more detials. * * ```javascript * // any context diff --git a/src/api/Errors.ts b/src/api/Errors.ts index 858f36118..a16b15bc9 100644 --- a/src/api/Errors.ts +++ b/src/api/Errors.ts @@ -73,6 +73,9 @@ export enum ResolveError { /** Returned if a call to one of the `raiseIntent` functions is made with an invalid context argument. Contexts should be Objects with at least a `type` field that has a `string` value.*/ MalformedContext = "MalformedContext", + /** Returned if `fdc3.addIntentListener` is called for a specified intent that the application has already added a listener for and has not subsequently removed it. */ + IntentListenerConflict = "IntentListenerConflict", + /** @experimental Returned if the specified Desktop Agent is not found, via a connected Desktop Agent Bridge.*/ DesktopAgentNotFound = "DesktopAgentNotFound", } diff --git a/toolbox/fdc3-conformance/App-Channel-Tests.md b/toolbox/fdc3-conformance/App-Channel-Tests.md index 597d0bc28..4e977bbdf 100644 --- a/toolbox/fdc3-conformance/App-Channel-Tests.md +++ b/toolbox/fdc3-conformance/App-Channel-Tests.md @@ -28,7 +28,7 @@ | App | Step | Details | |-----|--------------------|-----------------------------------------------------------------| | A | 1.Retrieve `Channel` |Retrieve a `Channel` object representing an 'App' channel called `test-channel` using:
`const testChannel = await fdc3.getOrCreateChannel("test-channel")` | -| A | 2.Add Context Listener |Add an _typed_ context listener for `fdc3.instrument`, using:
![2.0](https://img.shields.io/badge/FDC3-2.0-blue) `await testChannel.addContextListener("fdc3.instrument", handler)`
![1.2](https://img.shields.io/badge/FDC3-1.2-green) `testChannel.addContextListener("fdc3.instrument", handler)` +| A | 2.Add Context Listener |Add a _typed_ context listener for `fdc3.instrument`, using:
![2.0](https://img.shields.io/badge/FDC3-2.0-blue) `await testChannel.addContextListener("fdc3.instrument", handler)`
![1.2](https://img.shields.io/badge/FDC3-1.2-green) `testChannel.addContextListener("fdc3.instrument", handler)` | B | 3.Retrieve `Channel` |Retrieve a `Channel` object representing the same 'App' channel A did (`test-channel`)| | B | 4.Broadcast | B broadcasts both an `fdc3.instrument` context and an `fdc3.contact` context, using:
`testChannel.broadcast()`
`testChannel.broadcast()`| | A | 5.Receive Context | An fdc3.instrument context is received by the handler added in step 2.
Ensure that the fdc3.instrument received by A is identical to that sent by B
Ensure that the fdc3.contact context is NOT received. | @@ -52,3 +52,18 @@ - `ACContextHistoryTyped`: Perform above test. - `ACContextHistoryMultiple`: **B** Broadcasts multiple history items of both types. Ensure that only the last version of each type is received by **A**. - `ACContextHistoryLast`: In step 5. **A** retrieves the _untyped_ current context of the channel via `const currentContext = await testChannel.getCurrentContext()`. Ensure that A receives only the very last broadcast context item _of any type_. + + +## Multipe listeners On The Same Or Overlapping Ccontext types + +| App | Step | Details | +|-----|--------------------|----------------------------------------------------------------------------| +| A | 1.Retrieve `Channel` |Retrieve a `Channel` object representing an 'App' channel called `test-channel` using:
`const testChannel = await fdc3.getOrCreateChannel("test-channel")` | +| A | 2.Add Context Listener |Add an _untyped_ context listener to the channel, using:
![2.0](https://img.shields.io/badge/FDC3-2.0-blue) `await testChannel.addContextListener(null, handler1)`
![1.2](https://img.shields.io/badge/FDC3-1.2-green) `testChannel.addContextListener(null, handler1)` | +| A | 3.Add Context Listener |Add a _typed_ context listener for `fdc3.instrument` with a different handler, using:
![2.0](https://img.shields.io/badge/FDC3-2.0-blue) `await testChannel.addContextListener("fdc3.instrument", handler2)`
![1.2](https://img.shields.io/badge/FDC3-1.2-green) `testChannel.addContextListener("fdc3.instrument", handler2)`| +| B | 4.Retrieve `Channel` | Retrieve a `Channel` object representing the same 'App' channel A did (`test-channel`)| +| B | 5.Broadcast | Broadcast an `fdc3.instrument` Context to the channel with:
`testChannel.broadcast()`| +| A | 6.Receive Context | The handlers added in step 2 and 3 will receive the instrument context. Ensure that the instrument received by A is identical to that sent by B. | + +- ACMultipleOverlappingListeners1: Perform above test +- ACMultipleOverlappingListeners2: Perform above test, but instead of _untyped_ context listener, in step 2, use `fdc3.instrument` (handler should remain different) \ No newline at end of file diff --git a/toolbox/fdc3-conformance/User-Channel-Tests.md b/toolbox/fdc3-conformance/User-Channel-Tests.md index 923a5332c..d2640ede9 100644 --- a/toolbox/fdc3-conformance/User-Channel-Tests.md +++ b/toolbox/fdc3-conformance/User-Channel-Tests.md @@ -49,3 +49,17 @@ _NB: User Channels were called System Channels in FDC3 1.2. The new terminolog - `UCFilteredUsageUnsubscribe`: Perform above test, except that after joining, **A** then `unsubscribe()`s from the channel using the `listener.unsubscribe` function. Check that **A** does NOT receive anything. - `UCFilteredUsageLeave`: Perform above test, except that immediately after joining, **A** _leaves the channel_, and so receives nothing. - `UCFilteredUsageNoJoin`: Perform the above test, but skip step 2 so that **A** does NOT join a channel. Confirm that the _current channel_ for **A** is NOT set before continuing with the rest of the test. **A** should receive nothing. + + +## Broadcast With Multiple Listeners On The Same or Overlapping Types + +| App | Step | Details | +|-----|--------------------|-------------------------------------------------------------------------------------------------------------| +| A | 1.addContextListeners | A sets up two Context Listeners. One _untyped_ and one for `fdc3.contact` by calling: `addContextListener (null, handler1)`
`addContextListener ("fdc3.contact", handler2)`
![1.2](https://img.shields.io/badge/FDC3-1.2-green) A `Listener` object is returned for each.
![2.0](https://img.shields.io/badge/FDC3-2.0-blue) A promise resolving a `Listener` object is returned for each.
Check that this has an `unsubscribe` method for each. | +| A | 2.joinUserChannel |A joins the first available user channel using:
![1.2](https://img.shields.io/badge/FDC3-1.2-green) `getSystemChannels()` Check channels are returned.
![2.0](https://img.shields.io/badge/FDC3-2.0-blue) `getUserChannels()` Check **user** channels are returned.
Call `fdc3.joinChannel()` on the first non-global channel.| +| B | 3.joinUserChannel |B joins the same channel as A, via the same process in 2. | +| B | 4.Broadcast |`fdc3.broadcast()` . | +| A | 5.Receive Context | A's `fdc3.contact` object matches the one broadcast by B, both handlers from step 1 are triggered, and broadcast arrives on the correct listener. | + +- UCMultipleOverlappingListeners1: Perform above test +- UCMultipleOverlappingListeners2: Perform above test, but instead of _untyped_ context listener, in step 2, use `fdc3.instrument` (handler should remain different) \ No newline at end of file