From 541b5a736d94d25ebfbf4eeb79a5daa2f97880ad Mon Sep 17 00:00:00 2001 From: Bryan Ingle Date: Fri, 6 Dec 2024 10:24:56 -0700 Subject: [PATCH] Release. Bump version number --- package.json | 2 +- test/SpecRunner.js | 103 ++++++++++++++++++++++++++++++++++++++++----- 2 files changed, 94 insertions(+), 11 deletions(-) diff --git a/package.json b/package.json index 5bd53e9..2d7710c 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "@barchart/common-js", - "version": "4.46.1", + "version": "4.47.0", "description": "Library of common JavaScript utilities", "author": { "name": "Bryan Ingle", diff --git a/test/SpecRunner.js b/test/SpecRunner.js index 4875997..a39cfe2 100644 --- a/test/SpecRunner.js +++ b/test/SpecRunner.js @@ -4027,7 +4027,8 @@ module.exports = (() => { })(); },{"./assert":33}],27:[function(require,module,exports){ -const assert = require('./assert'); +const assert = require('./assert'), + is = require('./is'); module.exports = (() => { 'use strict'; @@ -4042,19 +4043,25 @@ module.exports = (() => { * @interface * @param {String} code - The unique code of the enumeration item. * @param {String} description - A description of the enumeration item. + * @param {Number=} mapping - An alternate key value (used when external systems identify enumeration items using integer values). */ class Enum { - constructor(code, description) { + constructor(code, description, mapping) { assert.argumentIsRequired(code, 'code', String); assert.argumentIsRequired(description, 'description', String); + assert.argumentIsOptional(mapping, 'mapping', Number); + if (is.number(mapping)) { + assert.argumentIsValid(mapping, 'mapping', is.integer, 'must be an integer'); + } this._code = code; this._description = description; + this._mapping = mapping || null; const c = this.constructor; if (!types.has(c)) { types.set(c, []); } - const existing = Enum.fromCode(c, code); - if (existing === null) { + const valid = Enum.fromCode(c, this._code) === null && (this._mapping === null || Enum.fromMapping(c, this._mapping) === null); + if (valid) { types.get(c).push(this); } } @@ -4079,6 +4086,17 @@ module.exports = (() => { return this._description; } + /** + * An alternate key value (used when external systems identify enumeration items + * using numeric values). This value will not be present for all enumerations. + * + * @public + * @returns {Number|null} + */ + get mapping() { + return this._mapping; + } + /** * Returns true if the provided {@link Enum} argument is equal * to the instance. @@ -4109,14 +4127,31 @@ module.exports = (() => { * @static * @param {Function} type - The enumeration type. * @param {String} code - The enumeration item's code. - * @returns {*|null} + * @returns {Enum|null} */ static fromCode(type, code) { return Enum.getItems(type).find(x => x.code === code) || null; } /** - * Returns all of the enumeration's items (given an enumeration type). + * Looks up a enumeration item; given the enumeration type and the enumeration + * item's value. If no matching item can be found, a null value is returned. + * + * @public + * @static + * @param {Function} type - The enumeration type. + * @param {String} mapping - The enumeration item's mapping value. + * @returns {Enum|null} + */ + static fromMapping(type, mapping) { + if (mapping === null) { + return null; + } + return Enum.getItems(type).find(x => x.mapping === mapping) || null; + } + + /** + * Returns the enumeration's items (given an enumeration type). * * @public * @static @@ -4133,7 +4168,7 @@ module.exports = (() => { return Enum; })(); -},{"./assert":33}],28:[function(require,module,exports){ +},{"./assert":33,"./is":40}],28:[function(require,module,exports){ const assert = require('./assert'), is = require('./is'); const Decimal = require('./Decimal'), @@ -21030,12 +21065,60 @@ describe('When Enum is extended (as types EnumA and EnumB) and type items are ad expect(Enum.fromCode(EnumB, 'y')).toBe(by); }); describe('and a duplicate item (A-x) is added', () => { - let axx = new EnumA('x', 'A-XX'); - it('should be still find the original instance in EnumA for X', () => { + let invalid = new EnumA('x', 'A-XX'); + it('should still only have two items', () => { + expect(Enum.getItems(EnumA).length).toEqual(2); + }); + it('should still able to find the original instance in EnumA for X', () => { expect(Enum.fromCode(EnumA, 'x')).toBe(ax); }); + it('should not be able to find the mapping for the duplicated item', () => { + expect(Enum.getItems(EnumA).some(x => x === invalid)).toEqual(false); + }); it('should should equal the original instance (for X)', () => { - expect(Enum.fromCode(EnumA, 'x').equals(axx)).toBe(true); + expect(Enum.fromCode(EnumA, 'x').equals(ax)).toBe(true); + }); + }); +}); +describe('When Enum is extended (as types EnumA and EnumB) and type items are added to each (X and Y) which include mapping values', () => { + 'use strict'; + + class EnumA extends Enum { + constructor(code, description, mapping) { + super(code, description, mapping); + } + } + class EnumB extends Enum { + constructor(code, description, mapping) { + super(code, description, mapping); + } + } + let ax = new EnumA('x', 'A-X', 1); + let ay = new EnumA('y', 'A-Y', 2); + let bx = new EnumB('x', 'B-X', 1); + let by = new EnumB('y', 'B-Y', 2); + it('should be able to find X in EnumA using the mapping value', () => { + expect(Enum.fromMapping(EnumA, 1)).toBe(ax); + }); + it('should be able to find Y in EnumA using the mapping value', () => { + expect(Enum.fromMapping(EnumA, 2)).toBe(ay); + }); + it('should be able to find X in EnumB using the mapping value', () => { + expect(Enum.fromMapping(EnumB, 1)).toBe(bx); + }); + it('should be able to find Y in EnumB using the mapping value', () => { + expect(Enum.fromMapping(EnumB, 2)).toBe(by); + }); + describe('and a duplicate mapping value is added', () => { + let invalid = new EnumA('z', 'A-Z', 2); + it('should still only have two items', () => { + expect(Enum.getItems(EnumA).length).toEqual(2); + }); + it('should still able to find the original instance in EnumA for Y', () => { + expect(Enum.fromMapping(EnumA, 2)).toBe(ay); + }); + it('should not be able to find the mapping for the duplicated item', () => { + expect(Enum.getItems(EnumA).some(x => x === invalid)).toEqual(false); }); }); });