From ca03d97abfc4f7176621d3846b41cf87bd27615a Mon Sep 17 00:00:00 2001 From: Momo Kornher Date: Tue, 7 Nov 2023 13:24:04 +0000 Subject: [PATCH] feat(jsii-reflect): TypeSystem can be locked to improve reflection performance Memoizes additional calls when the typesystem has been locked. --- packages/jsii-reflect/lib/_memoized.ts | 67 ++++++++++++----- packages/jsii-reflect/lib/class.ts | 14 +++- packages/jsii-reflect/lib/method.ts | 2 + packages/jsii-reflect/lib/type-system.ts | 19 +++++ packages/jsii-reflect/lib/type.ts | 2 +- packages/jsii-reflect/test/_memoized.test.ts | 79 ++++++++++++++++++++ 6 files changed, 163 insertions(+), 20 deletions(-) create mode 100644 packages/jsii-reflect/test/_memoized.test.ts diff --git a/packages/jsii-reflect/lib/_memoized.ts b/packages/jsii-reflect/lib/_memoized.ts index 3cc46383ed..9bce919802 100644 --- a/packages/jsii-reflect/lib/_memoized.ts +++ b/packages/jsii-reflect/lib/_memoized.ts @@ -1,7 +1,30 @@ -/* eslint-disable @typescript-eslint/ban-types -- WeakMap demands T extends object */ +import { TypeSystem } from './type-system'; +/* eslint-disable @typescript-eslint/ban-types -- WeakMap demands T extends object */ const CACHE = new WeakMap>(); +function memoizedGet(original: () => any, propertyKey: string): () => any { + return function (this: object): unknown { + let cache = CACHE.get(this); + if (cache == null) { + cache = new Map(); + CACHE.set(this, cache); + } + if (cache.has(propertyKey)) { + const result = cache.get(propertyKey); + if (Array.isArray(result)) { + // Return a copy of arrays as a precaution + return Array.from(result); + } + return result; + } + const result = original.call(this); + // If the result is an array, memoize a copy for safety. + cache.set(propertyKey, Array.isArray(result) ? Array.from(result) : result); + return result; + }; +} + /** * Decorates property readers for readonly properties so that their results are * memoized in a `WeakMap`-based cache. Those properties will consequently be @@ -9,6 +32,10 @@ const CACHE = new WeakMap>(); * * This can only be applied to property accessors (`public get foo(): any`), and not to * property declarations (`public readonly foo: any`). + * + * This should not be applied to any computations relying on a typesystem. + * The typesystem can be changed and thus change the result of the call. + * Use `memoizedWhenLocked` instead. */ export function memoized( _prototype: unknown, @@ -23,23 +50,27 @@ export function memoized( } const original = descriptor.get; - descriptor.get = function memoizedGet(this: object): unknown { - let cache = CACHE.get(this); - if (cache == null) { - cache = new Map(); - CACHE.set(this, cache); - } - if (cache.has(propertyKey)) { - const result = cache.get(propertyKey); - if (Array.isArray(result)) { - // Return a copy of arrays as a precaution - return Array.from(result); - } - return result; + descriptor.get = memoizedGet(original, propertyKey); +} + +export function memoizedWhenLocked( + _prototype: T, + propertyKey: string, + descriptor: PropertyDescriptor, +): void { + if (!descriptor.get) { + throw new Error(`@memoized can only be applied to property getters!`); + } + if (descriptor.set) { + throw new Error(`@memoized can only be applied to readonly properties!`); + } + + const original = descriptor.get; + descriptor.get = function (this: T): unknown { + if (this.system.isLocked) { + return memoizedGet(original, propertyKey).call(this); } - const result = original.call(this); - // If the result is an array, memoize a copy for safety. - cache.set(propertyKey, Array.isArray(result) ? Array.from(result) : result); - return result; + + return original.call(this); }; } diff --git a/packages/jsii-reflect/lib/class.ts b/packages/jsii-reflect/lib/class.ts index 21a8d703d8..866ae8ab83 100644 --- a/packages/jsii-reflect/lib/class.ts +++ b/packages/jsii-reflect/lib/class.ts @@ -1,5 +1,6 @@ import * as jsii from '@jsii/spec'; +import { memoizedWhenLocked } from './_memoized'; import { Assembly } from './assembly'; import { Initializer } from './initializer'; import { InterfaceType } from './interface'; @@ -20,6 +21,7 @@ export class ClassType extends ReferenceType { /** * Base class (optional). */ + @memoizedWhenLocked public get base(): ClassType | undefined { if (!this.spec.base) { return undefined; @@ -60,12 +62,22 @@ export class ClassType extends ReferenceType { /** * Returns list of all base classes (first is the direct base and last is the top-most). + * + * @deprecated use ClassType.ancestors instead */ public getAncestors() { + return this.ancestors; + } + + /** + * Returns list of all base classes (first is the direct base and last is the top-most). + */ + @memoizedWhenLocked + public get ancestors() { const out = new Array(); if (this.base) { out.push(this.base); - out.push(...this.base.getAncestors()); + out.push(...this.base.ancestors); } return out; } diff --git a/packages/jsii-reflect/lib/method.ts b/packages/jsii-reflect/lib/method.ts index 88f326a0fc..78bfad12d8 100644 --- a/packages/jsii-reflect/lib/method.ts +++ b/packages/jsii-reflect/lib/method.ts @@ -1,5 +1,6 @@ import * as jsii from '@jsii/spec'; +import { memoizedWhenLocked } from './_memoized'; import { Assembly } from './assembly'; import { Callable } from './callable'; import { Documentable } from './docs'; @@ -42,6 +43,7 @@ export class Method return this.spec.name; } + @memoizedWhenLocked public get overrides(): Type | undefined { if (!this.spec.overrides) { return undefined; diff --git a/packages/jsii-reflect/lib/type-system.ts b/packages/jsii-reflect/lib/type-system.ts index 6855924fb6..965a9d301a 100644 --- a/packages/jsii-reflect/lib/type-system.ts +++ b/packages/jsii-reflect/lib/type-system.ts @@ -22,6 +22,11 @@ export class TypeSystem { private readonly _cachedClasses = new Map(); + private _locked = false; + public get isLocked(): boolean { + return this._locked; + } + /** * All assemblies in this type system. */ @@ -29,6 +34,16 @@ export class TypeSystem { return Array.from(this._assemblyLookup.values()); } + /** + * Locks the TypeSystem from further changes + * + * Call this once all assemblies have been loaded. + * This allows the reflection to optimize and cache certain expensive calls. + */ + public lock() { + this._locked = true; + } + /** * Load all JSII dependencies of the given NPM package directory. * @@ -164,6 +179,10 @@ export class TypeSystem { } public addAssembly(asm: Assembly, options: { isRoot?: boolean } = {}) { + if (this.isLocked) { + throw new Error('The typesystem has been locked from further changes'); + } + if (asm.system !== this) { throw new Error('Assembly has been created for different typesystem'); } diff --git a/packages/jsii-reflect/lib/type.ts b/packages/jsii-reflect/lib/type.ts index 80d0841e3d..a05722136f 100644 --- a/packages/jsii-reflect/lib/type.ts +++ b/packages/jsii-reflect/lib/type.ts @@ -124,7 +124,7 @@ export abstract class Type implements Documentable, SourceLocatable { return this.getInterfaces(true).some((iface) => iface === base); } if (this.isClassType() && base.isClassType()) { - return this.getAncestors().some((clazz) => clazz === base); + return this.ancestors.some((clazz) => clazz === base); } return false; } diff --git a/packages/jsii-reflect/test/_memoized.test.ts b/packages/jsii-reflect/test/_memoized.test.ts new file mode 100644 index 0000000000..2dc69cb9e3 --- /dev/null +++ b/packages/jsii-reflect/test/_memoized.test.ts @@ -0,0 +1,79 @@ +import { TypeSystem } from '../lib'; +import { memoized, memoizedWhenLocked } from '../lib/_memoized'; + +const accessorSpy = jest.fn(() => 'foobar'); + +class TestClass { + public constructor(public readonly system: TypeSystem) {} + + public get uncached(): string { + return accessorSpy(); + } + + @memoized + public get cached(): string { + return accessorSpy(); + } + + @memoizedWhenLocked + public get cachedWhenLocked(): string { + return accessorSpy(); + } +} + +// eslint-disable-next-line @typescript-eslint/no-empty-function +function noop(_val: unknown) {} + +describe('memoized', () => { + beforeEach(() => { + accessorSpy.mockClear(); + }); + const subject = new TestClass(new TypeSystem()); + + test('cached property is memoized', () => { + // Access the property twice + noop(subject.cached); + noop(subject.cached); + + expect(accessorSpy).toHaveBeenCalledTimes(1); + expect(subject.cached).toBe('foobar'); + }); + + test('uncached property is not memoized', () => { + // Access the property twice + noop(subject.uncached); + noop(subject.uncached); + + expect(accessorSpy).toHaveBeenCalledTimes(2); + expect(subject.uncached).toBe('foobar'); + }); +}); + +describe('memoizedWhenLocked', () => { + let subject: TestClass; + beforeEach(() => { + accessorSpy.mockClear(); + subject = new TestClass(new TypeSystem()); + }); + + test('property is memoized when the typesystem is locked', () => { + // Lock the typesystem to enable memoizing + subject.system.lock(); + + // Access the property twice + noop(subject.cachedWhenLocked); + noop(subject.cachedWhenLocked); + + expect(accessorSpy).toHaveBeenCalledTimes(1); + expect(subject.cachedWhenLocked).toBe('foobar'); + }); + + test('property is not memoized when the typesystem is not locked', () => { + // Access the property twice + noop(subject.cachedWhenLocked); + noop(subject.cachedWhenLocked); + + expect(accessorSpy).toHaveBeenCalledTimes(2); + expect(subject.cachedWhenLocked).toBe('foobar'); + }); +});