Files
lodash/test/methodOf.spec.ts
2023-09-16 14:47:50 -07:00

147 lines
4.6 KiB
TypeScript

import assert from 'node:assert';
import lodashStable from 'lodash';
import { stubOne, _, stubTwo, stubThree, stubFour, noop, slice } from './utils';
import constant from '../src/constant';
describe('methodOf', () => {
it('should create a function that calls a method of a given key', () => {
const object = { a: stubOne };
lodashStable.each(['a', ['a']], (path) => {
const methodOf = _.methodOf(object);
assert.strictEqual(methodOf.length, 1);
assert.strictEqual(methodOf(path), 1);
});
});
it('should work with deep property values', () => {
const object = { a: { b: stubTwo } };
lodashStable.each(['a.b', ['a', 'b']], (path) => {
const methodOf = _.methodOf(object);
assert.strictEqual(methodOf(path), 2);
});
});
it('should work with a non-string `path`', () => {
const array = lodashStable.times(3, constant);
lodashStable.each([1, [1]], (path) => {
const methodOf = _.methodOf(array);
assert.strictEqual(methodOf(path), 1);
});
});
it('should coerce `path` to a string', () => {
function fn() {}
fn.toString = lodashStable.constant('fn');
const expected = [1, 2, 3, 4],
object = {
null: stubOne,
undefined: stubTwo,
fn: stubThree,
'[object Object]': stubFour,
},
paths = [null, undefined, fn, {}];
lodashStable.times(2, (index) => {
const actual = lodashStable.map(paths, (path) => {
const methodOf = _.methodOf(object);
return methodOf(index ? [path] : path);
});
assert.deepStrictEqual(actual, expected);
});
});
it('should work with inherited property values', () => {
function Foo() {}
Foo.prototype.a = stubOne;
lodashStable.each(['a', ['a']], (path) => {
const methodOf = _.methodOf(new Foo());
assert.strictEqual(methodOf(path), 1);
});
});
it('should use a key over a path', () => {
const object = { 'a.b': stubOne, a: { b: stubTwo } };
lodashStable.each(['a.b', ['a.b']], (path) => {
const methodOf = _.methodOf(object);
assert.strictEqual(methodOf(path), 1);
});
});
it('should return `undefined` when `object` is nullish', () => {
const values = [, null, undefined],
expected = lodashStable.map(values, noop);
lodashStable.each(['constructor', ['constructor']], (path) => {
const actual = lodashStable.map(values, (value, index) => {
const methodOf = index ? _.methodOf() : _.methodOf(value);
return methodOf(path);
});
assert.deepStrictEqual(actual, expected);
});
});
it('should return `undefined` for deep paths when `object` is nullish', () => {
const values = [, null, undefined],
expected = lodashStable.map(values, noop);
lodashStable.each(
['constructor.prototype.valueOf', ['constructor', 'prototype', 'valueOf']],
(path) => {
const actual = lodashStable.map(values, (value, index) => {
const methodOf = index ? _.methodOf() : _.methodOf(value);
return methodOf(path);
});
assert.deepStrictEqual(actual, expected);
},
);
});
it('should return `undefined` if parts of `path` are missing', () => {
const object = {},
methodOf = _.methodOf(object);
lodashStable.each(['a', 'a[1].b.c', ['a'], ['a', '1', 'b', 'c']], (path) => {
assert.strictEqual(methodOf(path), undefined);
});
});
it('should apply partial arguments to function', () => {
const object = {
fn: function () {
return slice.call(arguments);
},
};
const methodOf = _.methodOf(object, 1, 2, 3);
lodashStable.each(['fn', ['fn']], (path) => {
assert.deepStrictEqual(methodOf(path), [1, 2, 3]);
});
});
it('should invoke deep property methods with the correct `this` binding', () => {
const object = {
a: {
b: function () {
return this.c;
},
c: 1,
},
},
methodOf = _.methodOf(object);
lodashStable.each(['a.b', ['a', 'b']], (path) => {
assert.strictEqual(methodOf(path), 1);
});
});
});