From f4113d8e554dca564b33700175b572540e689508 Mon Sep 17 00:00:00 2001 From: Andrew Bulat Date: Tue, 19 Nov 2024 03:43:17 +0000 Subject: [PATCH 1/3] Add support for user-provided typings for the LiveObjects data structure Resolves DTP-963 --- src/plugins/liveobjects/livemap.ts | 22 ++++++++++++---------- src/plugins/liveobjects/liveobjects.ts | 10 ++++++++-- src/plugins/liveobjects/typings.ts | 22 ++++++++++++++++++++++ 3 files changed, 42 insertions(+), 12 deletions(-) create mode 100644 src/plugins/liveobjects/typings.ts diff --git a/src/plugins/liveobjects/livemap.ts b/src/plugins/liveobjects/livemap.ts index b29654c02..d20d1c47f 100644 --- a/src/plugins/liveobjects/livemap.ts +++ b/src/plugins/liveobjects/livemap.ts @@ -13,6 +13,7 @@ import { StateValue, } from './statemessage'; import { DefaultTimeserial, Timeserial } from './timeserial'; +import { LiveMapType } from './typings'; export interface ObjectIdStateData { /** A reference to another state object, used to support composable state objects. */ @@ -45,7 +46,7 @@ export interface LiveMapUpdate extends LiveObjectUpdate { update: { [keyName: string]: 'updated' | 'removed' }; } -export class LiveMap extends LiveObject { +export class LiveMap extends LiveObject { constructor( liveObjects: LiveObjects, private _semantics: MapSemantics, @@ -59,8 +60,8 @@ export class LiveMap extends LiveObject { * * @internal */ - static zeroValue(liveobjects: LiveObjects, objectId: string): LiveMap { - return new LiveMap(liveobjects, MapSemantics.LWW, objectId); + static zeroValue(liveobjects: LiveObjects, objectId: string): LiveMap { + return new LiveMap(liveobjects, MapSemantics.LWW, objectId); } /** @@ -69,8 +70,8 @@ export class LiveMap extends LiveObject { * * @internal */ - static fromStateObject(liveobjects: LiveObjects, stateObject: StateObject): LiveMap { - const obj = new LiveMap(liveobjects, stateObject.map?.semantics!, stateObject.objectId); + static fromStateObject(liveobjects: LiveObjects, stateObject: StateObject): LiveMap { + const obj = new LiveMap(liveobjects, stateObject.map?.semantics!, stateObject.objectId); obj.overrideWithStateObject(stateObject); return obj; } @@ -82,24 +83,25 @@ export class LiveMap extends LiveObject { * then you will get a reference to that Live Object if it exists in the local pool, or undefined otherwise. * If the value is not an objectId, then you will get that value. */ - get(key: string): LiveObject | StateValue | undefined { + // force the key to be of type string as we only allow strings as key in a map + get(key: TKey): T[TKey] { const element = this._dataRef.data.get(key); if (element === undefined) { - return undefined; + return undefined as T[TKey]; } if (element.tombstone === true) { - return undefined; + return undefined as T[TKey]; } // data exists for non-tombstoned elements const data = element.data!; if ('value' in data) { - return data.value; + return data.value as T[TKey]; } else { - return this._liveObjects.getPool().get(data.objectId); + return this._liveObjects.getPool().get(data.objectId) as T[TKey]; } } diff --git a/src/plugins/liveobjects/liveobjects.ts b/src/plugins/liveobjects/liveobjects.ts index 5ba586fe4..741544d55 100644 --- a/src/plugins/liveobjects/liveobjects.ts +++ b/src/plugins/liveobjects/liveobjects.ts @@ -8,6 +8,7 @@ import { LiveObject, LiveObjectUpdate } from './liveobject'; import { LiveObjectsPool, ROOT_OBJECT_ID } from './liveobjectspool'; import { StateMessage } from './statemessage'; import { SyncLiveObjectsDataPool } from './syncliveobjectsdatapool'; +import { DefaultRoot, LiveMapType } from './typings'; enum LiveObjectsEvents { SyncCompleted = 'SyncCompleted', @@ -36,13 +37,18 @@ export class LiveObjects { this._bufferedStateOperations = []; } - async getRoot(): Promise { + /** + * When called without a type variable, we return a default root type which is based on globally defined LiveObjects interface. + * A user can provide an explicit type for the getRoot method to explicitly set the LiveObjects type structure on this particular channel. + * This is useful when working with LiveObjects on multiple channels with different underlying data. + */ + async getRoot(): Promise> { // SYNC is currently in progress, wait for SYNC sequence to finish if (this._syncInProgress) { await this._eventEmitter.once(LiveObjectsEvents.SyncCompleted); } - return this._liveObjectsPool.get(ROOT_OBJECT_ID) as LiveMap; + return this._liveObjectsPool.get(ROOT_OBJECT_ID) as LiveMap; } /** diff --git a/src/plugins/liveobjects/typings.ts b/src/plugins/liveobjects/typings.ts new file mode 100644 index 000000000..879b6a29c --- /dev/null +++ b/src/plugins/liveobjects/typings.ts @@ -0,0 +1,22 @@ +import { LiveCounter } from './livecounter'; +import { LiveMap } from './livemap'; +import { StateValue } from './statemessage'; + +declare global { + // define a global interface which can be used by users to define their own types for LiveObjects. + export interface LiveObjectsTypes { + [key: string]: unknown; + } +} + +// LiveMap type representation of how it looks to the end-user. A mapping of string keys to the scalar values (StateValue) or other Live Objects. +export type LiveMapType = { [key: string]: StateValue | LiveMap | LiveCounter | undefined }; + +export type DefaultRoot = + // we need a way to understand when no types were provided by the user. + // we expect a "root" property to be set on LiveObjectsTypes interface, e.g. it won't be "unknown" anymore + unknown extends LiveObjectsTypes['root'] + ? LiveMapType // no types provided by the user, use the default map type for the root + : LiveObjectsTypes['root'] extends LiveMapType + ? LiveObjectsTypes['root'] // "root" was provided by the user, and it is of an expected type, we can use this interface for the root object in LiveObjects. + : `Provided type definition for the "root" object in LiveObjectsTypes is not of an expected LiveMapType type`; From 73c8abcbda0261392f754e6fb88821a52e0d7a56 Mon Sep 17 00:00:00 2001 From: Andrew Bulat Date: Tue, 19 Nov 2024 04:07:41 +0000 Subject: [PATCH 2/3] Move types required for user-provided LiveObjects typings to ably.d.ts Add globally defined `LiveObjectsTypes` to `intentionallyNotExported` [1] list for `typedoc`, as typedocs does not include documentation for the globally defined interfaces and complains that "__global.LiveObjectsTypes, defined in ./ably.d.ts, is referenced by ably.DefaultRoot but not included in the documentation." [1] https://typedoc.org/documents/Options.Validation.html#intentionallynotexported --- ably.d.ts | 57 ++++++++++++++++++++++++-- src/plugins/liveobjects/livemap.ts | 8 ++-- src/plugins/liveobjects/liveobjects.ts | 3 +- src/plugins/liveobjects/typings.ts | 22 ---------- typedoc.json | 3 +- 5 files changed, 61 insertions(+), 32 deletions(-) delete mode 100644 src/plugins/liveobjects/typings.ts diff --git a/ably.d.ts b/ably.d.ts index 225fc806c..5f375b1ec 100644 --- a/ably.d.ts +++ b/ably.d.ts @@ -2039,25 +2039,76 @@ export declare interface LiveObjects { /** * Retrieves the root {@link LiveMap} object for state on a channel. * + * A type parameter can be provided to describe the structure of the LiveObjects state on the channel. By default, it uses types from the globally defined `LiveObjectsTypes` interface. + * + * You can specify custom types for LiveObjects by defining a global `LiveObjectsTypes` interface with a `root` property that conforms to {@link LiveMapType}. + * + * Example: + * + * ```typescript + * import { LiveCounter } from 'ably'; + * + * type MyRoot = { + * myTypedKey: LiveCounter; + * }; + * + * declare global { + * export interface LiveObjectsTypes { + * root: MyRoot; + * } + * } + * ``` + * * @returns A promise which, upon success, will be fulfilled with a {@link LiveMap} object. Upon failure, the promise will be rejected with an {@link ErrorInfo} object which explains the error. */ - getRoot(): Promise; + getRoot(): Promise>; } +declare global { + /** + * A globally defined interface that allows users to define custom types for LiveObjects. + */ + export interface LiveObjectsTypes { + [key: string]: unknown; + } +} + +/** + * Represents the type of data stored in a {@link LiveMap}. + * It maps string keys to scalar values ({@link StateValue}), or other LiveObjects. + */ +export type LiveMapType = { [key: string]: StateValue | LiveMap | LiveCounter | undefined }; + +/** + * The default type for the `root` object in the LiveObjects, based on the globally defined {@link LiveObjectsTypes} interface. + * + * - If no custom types are provided in `LiveObjectsTypes`, defaults to an untyped root map representation using the {@link LiveMapType} interface. + * - If a `root` type exists in `LiveObjectsTypes` and conforms to the {@link LiveMapType} interface, it is used as the type for the `root` object. + * - If the provided `root` type does not match {@link LiveMapType}, a type error message is returned. + */ +export type DefaultRoot = + // we need a way to know when no types were provided by the user. + // we expect a "root" property to be set on LiveObjectsTypes interface, e.g. it won't be "unknown" anymore + unknown extends LiveObjectsTypes['root'] + ? LiveMapType // no custom types provided; use the default untyped map representation for the root + : LiveObjectsTypes['root'] extends LiveMapType + ? LiveObjectsTypes['root'] // "root" property exists, and it is of an expected type, we can use this interface for the root object in LiveObjects. + : `Provided type definition for the "root" object in LiveObjectsTypes is not of an expected LiveMapType`; + /** * The `LiveMap` class represents a key/value map data structure, similar to a JavaScript Map, where all changes are synchronized across clients in realtime. * Conflict-free resolution for updates follows Last Write Wins (LWW) semantics, meaning that if two clients update the same key in the map, the update with the most recent timestamp wins. * * Keys must be strings. Values can be another Live Object, or a primitive type, such as a string, number, boolean, or binary data (see {@link StateValue}). */ -export declare interface LiveMap extends LiveObject { +export declare interface LiveMap extends LiveObject { /** * Returns the value associated with a given key. Returns `undefined` if the key doesn't exist in a map. * * @param key - The key to retrieve the value for. * @returns A {@link LiveObject}, a primitive type (string, number, boolean, or binary data) or `undefined` if the key doesn't exist in a map. */ - get(key: string): LiveObject | StateValue | undefined; + get(key: TKey): T[TKey]; /** * Returns the number of key/value pairs in the map. diff --git a/src/plugins/liveobjects/livemap.ts b/src/plugins/liveobjects/livemap.ts index d20d1c47f..07577d4b2 100644 --- a/src/plugins/liveobjects/livemap.ts +++ b/src/plugins/liveobjects/livemap.ts @@ -1,5 +1,6 @@ import deepEqual from 'deep-equal'; +import type * as API from '../../../ably'; import { LiveObject, LiveObjectData, LiveObjectUpdate, LiveObjectUpdateNoop } from './liveobject'; import { LiveObjects } from './liveobjects'; import { @@ -13,7 +14,6 @@ import { StateValue, } from './statemessage'; import { DefaultTimeserial, Timeserial } from './timeserial'; -import { LiveMapType } from './typings'; export interface ObjectIdStateData { /** A reference to another state object, used to support composable state objects. */ @@ -46,7 +46,7 @@ export interface LiveMapUpdate extends LiveObjectUpdate { update: { [keyName: string]: 'updated' | 'removed' }; } -export class LiveMap extends LiveObject { +export class LiveMap extends LiveObject { constructor( liveObjects: LiveObjects, private _semantics: MapSemantics, @@ -60,7 +60,7 @@ export class LiveMap extends LiveObject(liveobjects: LiveObjects, objectId: string): LiveMap { + static zeroValue(liveobjects: LiveObjects, objectId: string): LiveMap { return new LiveMap(liveobjects, MapSemantics.LWW, objectId); } @@ -70,7 +70,7 @@ export class LiveMap extends LiveObject(liveobjects: LiveObjects, stateObject: StateObject): LiveMap { + static fromStateObject(liveobjects: LiveObjects, stateObject: StateObject): LiveMap { const obj = new LiveMap(liveobjects, stateObject.map?.semantics!, stateObject.objectId); obj.overrideWithStateObject(stateObject); return obj; diff --git a/src/plugins/liveobjects/liveobjects.ts b/src/plugins/liveobjects/liveobjects.ts index 741544d55..75b743d5b 100644 --- a/src/plugins/liveobjects/liveobjects.ts +++ b/src/plugins/liveobjects/liveobjects.ts @@ -8,7 +8,6 @@ import { LiveObject, LiveObjectUpdate } from './liveobject'; import { LiveObjectsPool, ROOT_OBJECT_ID } from './liveobjectspool'; import { StateMessage } from './statemessage'; import { SyncLiveObjectsDataPool } from './syncliveobjectsdatapool'; -import { DefaultRoot, LiveMapType } from './typings'; enum LiveObjectsEvents { SyncCompleted = 'SyncCompleted', @@ -42,7 +41,7 @@ export class LiveObjects { * A user can provide an explicit type for the getRoot method to explicitly set the LiveObjects type structure on this particular channel. * This is useful when working with LiveObjects on multiple channels with different underlying data. */ - async getRoot(): Promise> { + async getRoot(): Promise> { // SYNC is currently in progress, wait for SYNC sequence to finish if (this._syncInProgress) { await this._eventEmitter.once(LiveObjectsEvents.SyncCompleted); diff --git a/src/plugins/liveobjects/typings.ts b/src/plugins/liveobjects/typings.ts deleted file mode 100644 index 879b6a29c..000000000 --- a/src/plugins/liveobjects/typings.ts +++ /dev/null @@ -1,22 +0,0 @@ -import { LiveCounter } from './livecounter'; -import { LiveMap } from './livemap'; -import { StateValue } from './statemessage'; - -declare global { - // define a global interface which can be used by users to define their own types for LiveObjects. - export interface LiveObjectsTypes { - [key: string]: unknown; - } -} - -// LiveMap type representation of how it looks to the end-user. A mapping of string keys to the scalar values (StateValue) or other Live Objects. -export type LiveMapType = { [key: string]: StateValue | LiveMap | LiveCounter | undefined }; - -export type DefaultRoot = - // we need a way to understand when no types were provided by the user. - // we expect a "root" property to be set on LiveObjectsTypes interface, e.g. it won't be "unknown" anymore - unknown extends LiveObjectsTypes['root'] - ? LiveMapType // no types provided by the user, use the default map type for the root - : LiveObjectsTypes['root'] extends LiveMapType - ? LiveObjectsTypes['root'] // "root" was provided by the user, and it is of an expected type, we can use this interface for the root object in LiveObjects. - : `Provided type definition for the "root" object in LiveObjectsTypes is not of an expected LiveMapType type`; diff --git a/typedoc.json b/typedoc.json index 24faf3bad..7c9c17476 100644 --- a/typedoc.json +++ b/typedoc.json @@ -20,5 +20,6 @@ "TypeAlias", "Variable", "Namespace" - ] + ], + "intentionallyNotExported": ["__global.LiveObjectsTypes"] } From 3938b639ecd726e90eee2dd49c6507e43a07a8bd Mon Sep 17 00:00:00 2001 From: Andrew Bulat Date: Tue, 19 Nov 2024 04:31:36 +0000 Subject: [PATCH 3/3] Add tests for user-provided LiveObjects types to the LiveObjects package test --- .../browser/template/src/ably.config.d.ts | 21 ++++++++++ .../browser/template/src/index-liveobjects.ts | 42 ++++++++++++++----- .../browser/template/src/tsconfig.json | 1 + 3 files changed, 54 insertions(+), 10 deletions(-) create mode 100644 test/package/browser/template/src/ably.config.d.ts diff --git a/test/package/browser/template/src/ably.config.d.ts b/test/package/browser/template/src/ably.config.d.ts new file mode 100644 index 000000000..e5bca7718 --- /dev/null +++ b/test/package/browser/template/src/ably.config.d.ts @@ -0,0 +1,21 @@ +import { LiveCounter, LiveMap } from 'ably'; + +type CustomRoot = { + numberKey: number; + stringKey: string; + booleanKey: boolean; + couldBeUndefined?: string; + mapKey?: LiveMap<{ + foo: 'bar'; + nestedMap?: LiveMap<{ + baz: 'qux'; + }>; + }>; + counterKey?: LiveCounter; +}; + +declare global { + export interface LiveObjectsTypes { + root: CustomRoot; + } +} diff --git a/test/package/browser/template/src/index-liveobjects.ts b/test/package/browser/template/src/index-liveobjects.ts index 9334aadf4..1cd27b021 100644 --- a/test/package/browser/template/src/index-liveobjects.ts +++ b/test/package/browser/template/src/index-liveobjects.ts @@ -1,7 +1,12 @@ import * as Ably from 'ably'; import LiveObjects from 'ably/liveobjects'; +import { CustomRoot } from './ably.config'; import { createSandboxAblyAPIKey } from './sandbox'; +type ExplicitRootType = { + someOtherKey: string; +}; + globalThis.testAblyPackage = async function () { const key = await createSandboxAblyAPIKey({ featureFlags: ['enableChannelState'] }); @@ -11,12 +16,27 @@ globalThis.testAblyPackage = async function () { // check liveObjects can be accessed const liveObjects = channel.liveObjects; await channel.attach(); - // root should be a LiveMap object - const root: Ably.LiveMap = await liveObjects.getRoot(); + // expect root to be a LiveMap instance with LiveObjects types defined via the global LiveObjectsTypes interface + // also checks that we can refer to the LiveObjects types exported from 'ably' by referencing a LiveMap interface + const root: Ably.LiveMap = await liveObjects.getRoot(); + + // check root has expected LiveMap TypeScript type methods + const size: number = root.size(); - // check root is recognized as LiveMap TypeScript type - root.get('someKey'); - root.size(); + // check custom user provided typings via LiveObjectsTypes are working: + // keys on a root: + const aNumber: number = root.get('numberKey'); + const aString: string = root.get('stringKey'); + const aBoolean: boolean = root.get('booleanKey'); + const couldBeUndefined: string | undefined = root.get('couldBeUndefined'); + // live objects on a root: + const counter: Ably.LiveCounter | undefined = root.get('counterKey'); + const map: LiveObjectsTypes['root']['mapKey'] = root.get('mapKey'); + // check string literal types works + // need to use nullish coalescing as we didn't actually create any data on the root, + // so the next calls would fail. we only need to check that TypeScript types work + const foo: 'bar' = map?.get('foo')!; + const baz: 'qux' = map?.get('nestedMap')?.get('baz')!; // check LiveMap subscription callback has correct TypeScript types const { unsubscribe } = root.subscribe(({ update }) => { @@ -31,13 +51,15 @@ globalThis.testAblyPackage = async function () { }); unsubscribe(); - // check LiveCounter types also behave as expected - const counter = root.get('randomKey') as Ably.LiveCounter | undefined; - // use nullish coalescing as we didn't actually create a counter object on the root, - // so the next calls would fail. we only need to check that TypeScript types work - const value: number = counter?.value(); + // check LiveCounter type also behaves as expected + // same deal with nullish coalescing + const value: number = counter?.value()!; const counterSubscribeResponse = counter?.subscribe(({ update }) => { const shouldBeANumber: number = update.inc; }); counterSubscribeResponse?.unsubscribe(); + + // check can provide custom types for the getRoot method, ignoring global LiveObjectsTypes interface + const explicitRoot: Ably.LiveMap = await liveObjects.getRoot(); + const someOtherKey: string = explicitRoot.get('someOtherKey'); }; diff --git a/test/package/browser/template/src/tsconfig.json b/test/package/browser/template/src/tsconfig.json index b206a6399..3230e8697 100644 --- a/test/package/browser/template/src/tsconfig.json +++ b/test/package/browser/template/src/tsconfig.json @@ -1,6 +1,7 @@ { "include": ["**/*.ts", "**/*.tsx"], "compilerOptions": { + "strictNullChecks": true, "resolveJsonModule": true, "esModuleInterop": true, "module": "esnext",