diff --git a/packages/utils/__tests__/index.test.js b/packages/utils/__tests__/index.test.js index a57334b021..304b89dba5 100644 --- a/packages/utils/__tests__/index.test.js +++ b/packages/utils/__tests__/index.test.js @@ -10,6 +10,7 @@ test('should match expected exports', () => { 'isNullish', 'isNullOrUndefined', 'isObject', + 'isPlainObject', 'isWhitespace', // dom @@ -36,6 +37,7 @@ test('should match expected exports', () => { 'callAll', 'callEventHandlers', 'dataAttr', + 'deepmerge', 'noop', 'once', 'runIfFn', diff --git a/packages/utils/src/__tests__/assertion.test.js b/packages/utils/src/__tests__/assertion.test.js index b842e588a9..fd64607880 100644 --- a/packages/utils/src/__tests__/assertion.test.js +++ b/packages/utils/src/__tests__/assertion.test.js @@ -1,4 +1,4 @@ -/* eslint-disable */ +import { runInNewContext } from 'node:vm'; import { isBlankString, isEmptyArray, @@ -7,6 +7,7 @@ import { isNullish, isNullOrUndefined, isObject, + isPlainObject, isWhitespace, noop, } from '@tonic-ui/utils/src'; @@ -34,7 +35,6 @@ describe('Check whether the value is a blank string', () => { describe('Check whether the value is an empty array', () => { it('should return true', () => { expect(isEmptyArray([])).toBe(true); - expect(isEmptyArray(new Array())).toBe(true); }); it('should return false', () => { @@ -47,12 +47,8 @@ describe('Check whether the value is an empty array', () => { expect(isEmptyArray(undefined)).toBe(false); expect(isEmptyArray('')).toBe(false); expect(isEmptyArray(' ')).toBe(false); - expect(isEmptyArray(new Boolean())).toBe(false); + expect(isEmptyArray(() => {})).toBe(false); expect(isEmptyArray(new Date())).toBe(false); - expect(isEmptyArray(new Function())).toBe(false); - expect(isEmptyArray(new Number())).toBe(false); - expect(isEmptyArray(new Object())).toBe(false); - expect(isEmptyArray(new String())).toBe(false); expect(isEmptyArray(new RegExp())).toBe(false); }); }); @@ -60,7 +56,6 @@ describe('Check whether the value is an empty array', () => { describe('Check whether the value is an empty object', () => { it('should return true', () => { expect(isEmptyObject({})).toBe(true); - expect(isEmptyObject(new Object())).toBe(true); }); it('should return false', () => { @@ -73,12 +68,8 @@ describe('Check whether the value is an empty object', () => { expect(isEmptyObject(undefined)).toBe(false); expect(isEmptyObject('')).toBe(false); expect(isEmptyObject(' ')).toBe(false); - expect(isEmptyObject(new Array())).toBe(false); - expect(isEmptyObject(new Boolean())).toBe(false); + expect(isEmptyObject(() => {})).toBe(false); expect(isEmptyObject(new Date())).toBe(false); - expect(isEmptyObject(new Function())).toBe(false); - expect(isEmptyObject(new Number())).toBe(false); - expect(isEmptyObject(new String())).toBe(false); expect(isEmptyObject(new RegExp())).toBe(false); }); }); @@ -124,7 +115,6 @@ describe('Check whether the value is an object', () => { it('should return true', () => { expect(isObject({})).toBe(true); expect(isObject(noop)).toBe(true); - expect(isObject(new Object())).toBe(true); }); it('should return false', () => { @@ -138,7 +128,52 @@ describe('Check whether the value is an object', () => { expect(isObject(' ')).toBe(false); }); }); - + +describe('Check whether the value is a plain object', () => { + function Foo(x) { + this.x = x; + } + + function ObjectConstructor() {} + ObjectConstructor.prototype.constructor = Object; + + it('should return true', () => { + expect(isPlainObject({})).toBe(true); + expect(isPlainObject({ foo: true })).toBe(true); + expect(isPlainObject({ constructor: Foo })).toBe(true); + expect(isPlainObject({ valueOf: 0 })).toBe(true); + expect(isPlainObject(Object.create(null))).toBe(true); + expect(isPlainObject(runInNewContext('({})'))).toBe(true); + }); + + it('should return false', () => { + expect(isPlainObject(['foo', 'bar'])).toBe(false); + expect(isPlainObject(new Foo(1))).toBe(false); + expect(isPlainObject(Math)).toBe(false); + expect(isPlainObject(JSON)).toBe(false); + expect(isPlainObject(Atomics)).toBe(false); // eslint-disable-line no-undef + expect(isPlainObject(Error)).toBe(false); + expect(isPlainObject(() => {})).toBe(false); + expect(isPlainObject(/./)).toBe(false); + expect(isPlainObject(null)).toBe(false); + expect(isPlainObject(undefined)).toBe(false); + expect(isPlainObject(Number.NaN)).toBe(false); + expect(isPlainObject('')).toBe(false); + expect(isPlainObject(0)).toBe(false); + expect(isPlainObject(false)).toBe(false); + expect(isPlainObject(new ObjectConstructor())).toBe(false); + expect(isPlainObject(Object.create({}))).toBe(false); + + (function () { + expect(isPlainObject(arguments)).toBe(false); // eslint-disable-line prefer-rest-params + }()); + + const foo = new Foo(); + foo.constructor = Object; + expect(isPlainObject(foo)).toBe(false); + }); +}); + describe('Check whether the value passed is all whitespace', () => { it('should return true', () => { expect(isWhitespace(' ')).toBe(true); diff --git a/packages/utils/src/__tests__/index.test.js b/packages/utils/src/__tests__/index.test.js index cb6d48ed8d..12ad63b8d1 100644 --- a/packages/utils/src/__tests__/index.test.js +++ b/packages/utils/src/__tests__/index.test.js @@ -11,6 +11,7 @@ test('should match expected exports', () => { 'isNullish', 'isNullOrUndefined', 'isObject', + 'isPlainObject', 'isWhitespace', // dom @@ -37,6 +38,7 @@ test('should match expected exports', () => { 'callAll', 'callEventHandlers', 'dataAttr', + 'deepmerge', 'noop', 'once', 'runIfFn', diff --git a/packages/utils/src/__tests__/shared.test.js b/packages/utils/src/__tests__/shared.test.js index 46156e13a4..8dfdbce5b5 100644 --- a/packages/utils/src/__tests__/shared.test.js +++ b/packages/utils/src/__tests__/shared.test.js @@ -1,8 +1,10 @@ +import { runInNewContext } from 'node:vm'; import { ariaAttr, callAll, callEventHandlers, dataAttr, + deepmerge, noop, once, runIfFn, @@ -14,19 +16,14 @@ afterEach(() => { jest.resetAllMocks(); }); -describe('ariaAttr / dataAttr', () => { - it('should render correct aria-* and data-* attributes', () => { - const ariaProps = { +describe('ariaAttr', () => { + it('should render correct aria-* attributes', () => { + const ariaAttrs = { 'aria-disabled': ariaAttr(true), - 'data-disabled': dataAttr(true), 'aria-selected': ariaAttr(false), - 'data-selected': dataAttr(false), }; - - expect(ariaProps['aria-disabled']).toBe(true); - expect(ariaProps['data-disabled']).toBe(''); - expect(ariaProps['aria-selected']).toBe(undefined); - expect(ariaProps['data-selected']).toBe(undefined); + expect(ariaAttrs['aria-disabled']).toBe(true); + expect(ariaAttrs['aria-selected']).toBe(undefined); }); }); @@ -78,6 +75,125 @@ describe('callEventHandlers', () => { }); }); +describe('dataAttr', () => { + it('should render correct data-* attributes', () => { + const dataAttrs = { + 'data-disabled': dataAttr(true), + 'data-selected': dataAttr(false), + }; + expect(dataAttrs['data-disabled']).toBe(''); + expect(dataAttrs['data-selected']).toBe(undefined); + }); +}); + +describe('deepmerge', () => { + it('should not be subject to prototype pollution via __proto__', () => { + const result = deepmerge( + {}, + JSON.parse('{ "myProperty": "a", "__proto__" : { "isAdmin" : true } }'), + { + clone: false, + } + ); + + expect(result.__proto__).toHaveProperty('isAdmin'); // eslint-disable-line no-proto + expect({}).not.toHaveProperty('isAdmin'); + }); + + it('should not be subject to prototype pollution via constructor', () => { + const result = deepmerge( + {}, + JSON.parse('{ "myProperty": "a", "constructor" : { "prototype": { "isAdmin" : true } } }'), + { + clone: true, + } + ); + + expect(result.constructor.prototype).toHaveProperty('isAdmin'); + expect({}).not.toHaveProperty('isAdmin'); + }); + + it('should not be subject to prototype pollution via prototype', () => { + const result = deepmerge( + {}, + JSON.parse('{ "myProperty": "a", "prototype": { "isAdmin" : true } }'), + { + clone: false, + } + ); + + expect(result.prototype).toHaveProperty('isAdmin'); + expect({}).not.toHaveProperty('isAdmin'); + }); + + it('should appropriately copy the fields without prototype pollution', () => { + const result = deepmerge( + {}, + JSON.parse('{ "myProperty": "a", "__proto__" : { "isAdmin" : true } }') + ); + + expect(result.__proto__).toHaveProperty('isAdmin'); // eslint-disable-line no-proto + expect({}).not.toHaveProperty('isAdmin'); + }); + + it('should merge objects across realms', function test() { + if (!/jsdom/.test(window.navigator.userAgent)) { + this.skip(); + } + + const vmObject = runInNewContext('({hello: "realm"})'); + const result = deepmerge({ hello: 'original' }, vmObject); + expect(result.hello).toBe('realm'); + }); + + it('should not merge HTML elements', () => { + const element = document.createElement('div'); + const element2 = document.createElement('div'); + + const result = deepmerge({ element }, { element: element2 }); + + expect(result.element).toBe(element2); + }); + + it('should reset source when target is undefined', () => { + const result = deepmerge( + { + '&.disabled': { + color: 'red', + }, + }, + { + '&.disabled': undefined, + } + ); + expect(result).toEqual({ + '&.disabled': undefined, + }); + }); + + it('should merge keys that do not exist in source', () => { + const result = deepmerge({ foo: { baz: 'test' } }, { foo: { bar: 'test' }, bar: 'test' }); + expect(result).toEqual({ + foo: { baz: 'test', bar: 'test' }, + bar: 'test', + }); + }); + + it('should deep clone source key object if target key does not exist', () => { + const foo = { foo: { baz: 'test' } }; + const bar = {}; + + const result = deepmerge(bar, foo); + + expect(result).toEqual({ foo: { baz: 'test' } }); + + result.foo.baz = 'new test'; + + expect(result).toEqual({ foo: { baz: 'new test' } }); + expect(foo).toEqual({ foo: { baz: 'test' } }); + }); +}); + describe('runIfFn', () => { it('should run function if function or else return value', () => { expect(runIfFn(() => 2)).toStrictEqual(2); diff --git a/packages/utils/src/assertion.js b/packages/utils/src/assertion.js index 16835760f6..bf93302ae6 100644 --- a/packages/utils/src/assertion.js +++ b/packages/utils/src/assertion.js @@ -30,6 +30,16 @@ export const isObject = (value) => { return !isNullish(value) && (typeof value === 'object' || typeof value === 'function') && !Array.isArray(value); }; +// https://github.com/sindresorhus/is-plain-obj/blob/main/index.js +export const isPlainObject = (value) => { + if (typeof value !== 'object' || value === null) { + return false; + } + + const prototype = Object.getPrototypeOf(value); + return (prototype === null || prototype === Object.prototype || Object.getPrototypeOf(prototype) === null) && !(Symbol.toStringTag in value) && !(Symbol.iterator in value); +}; + export const isWhitespace = (value) => { // @see https://github.com/jonschlinkert/whitespace-regex // eslint-disable-next-line no-control-regex diff --git a/packages/utils/src/shared.js b/packages/utils/src/shared.js index 24da7d2269..6b50f3a146 100644 --- a/packages/utils/src/shared.js +++ b/packages/utils/src/shared.js @@ -1,4 +1,5 @@ import { ensureArray, ensureBoolean, ensureString } from 'ensure-type'; +import { isPlainObject } from './assertion'; const _joinWords = (words) => { words = ensureArray(words); @@ -14,6 +15,20 @@ const _joinWords = (words) => { return `'${words.slice(0, -1).join('\', \'')}', and '${words.slice(-1)}'`; }; +const _deepClone = (source) => { + if (!isPlainObject(source)) { + return source; + } + + const output = {}; + + Object.keys(source).forEach((key) => { + output[key] = _deepClone(source[key]); + }); + + return output; +}; + export const ariaAttr = (condition) => { return ensureBoolean(condition) ? true : undefined; }; @@ -39,6 +54,28 @@ export const dataAttr = (condition) => { return condition ? '' : undefined; }; +export const deepmerge = (target, source, options = { clone: true }) => { + const output = options.clone ? { ...target } : target; + + if (isPlainObject(target) && isPlainObject(source)) { + Object.keys(source).forEach((key) => { + if ( + isPlainObject(source[key]) && + Object.prototype.hasOwnProperty.call(target, key) && + isPlainObject(target[key]) + ) { + output[key] = deepmerge(target[key], source[key], options); + } else if (options.clone) { + output[key] = isPlainObject(source[key]) ? _deepClone(source[key]) : source[key]; + } else { + output[key] = source[key]; + } + }); + } + + return output; +}; + export const noop = () => {}; export const once = (fn) => {