Files
alfresco-ng2-components/lib/js-api/test/lazy-api.spec.ts
Michal Kinas bc41c33ba8 [ACS-11205] Add LazyApi decorator (#11711)
* [ACS-11205] Add LazyApi decorator

* [ACS-11205] Add unit tests and CR fixes

* [ACS-11259] Storybook build fix
2026-03-12 10:15:38 +01:00

129 lines
3.8 KiB
TypeScript

/*!
* @license
* Copyright © 2005-2025 Hyland Software, Inc. and its affiliates. All rights reserved.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
import assert from 'assert';
import { LazyApi } from '../src/utils/lazy-api';
describe('LazyApi', () => {
it('should create a lazy-loaded property on the target prototype', () => {
class TestApi {
@LazyApi(() => ({ value: 13 }))
api: { value: number };
}
const instance = new TestApi();
assert.deepStrictEqual(instance.api, { value: 13 });
});
it('should cache the instance of the property after first access', () => {
let accessCount = 0;
class TestApi {
@LazyApi(() => {
accessCount++;
return { value: 'cached' };
})
api: { value: string };
}
const instance = new TestApi();
const first = instance.api;
const second = instance.api;
assert.strictEqual(first, second);
assert.strictEqual(accessCount, 1);
});
it('should pass the class instance as argument to the factory', () => {
let receivedSelf: any;
class TestApi {
config = 'test-config';
@LazyApi((self: any) => {
receivedSelf = self;
return { config: self.config };
})
api: { config: string };
}
const instance = new TestApi();
const result = instance.api;
assert.strictEqual(receivedSelf, instance);
assert.deepStrictEqual(result, { config: 'test-config' });
});
it('should create separate cached instances per class instance', () => {
let accessCount = 0;
class TestApi {
@LazyApi(() => {
accessCount++;
return { id: accessCount };
})
api: { id: number };
}
const instance1 = new TestApi();
const instance2 = new TestApi();
assert.strictEqual(instance1.api.id, 1);
assert.strictEqual(instance2.api.id, 2);
assert.strictEqual(accessCount, 2);
});
it('should define the property as enumerable and configurable', () => {
class TestApi {
@LazyApi(() => 'value')
api: string;
}
const descriptor = Object.getOwnPropertyDescriptor(TestApi.prototype, 'api');
assert.strictEqual(descriptor.enumerable, true);
assert.strictEqual(descriptor.configurable, true);
});
it('should support multiple decorated properties on the same class', () => {
class TestApi {
@LazyApi(() => 'first')
api1: string;
@LazyApi(() => 'second')
api2: string;
}
const instance = new TestApi();
assert.strictEqual(instance.api1, 'first');
assert.strictEqual(instance.api2, 'second');
});
it('should store cached value under _propertyKey on the instance', () => {
class TestApi {
@LazyApi(() => 'cached-value')
myApi: string;
}
const instance = new TestApi();
assert.strictEqual(instance['_myApi'], undefined);
instance.myApi;
assert.strictEqual(instance['_myApi'], 'cached-value');
});
});