Files
lodash/test/transform.spec.ts
2023-09-16 16:18:43 -07:00

207 lines
6.5 KiB
TypeScript

import assert from 'node:assert';
import lodashStable from 'lodash';
import {
stubTrue,
square,
typedArrays,
noop,
stubObject,
stubFalse,
falsey,
slice,
realm,
} from './utils';
import transform from '../src/transform';
describe('transform', () => {
function Foo() {
this.a = 1;
this.b = 2;
this.c = 3;
}
it('should create an object with the same `[[Prototype]]` as `object` when `accumulator` is nullish', () => {
let accumulators = [, null, undefined],
object = new Foo(),
expected = lodashStable.map(accumulators, stubTrue);
const iteratee = function (result, value, key) {
result[key] = square(value);
};
const mapper = function (accumulator, index) {
return index ? transform(object, iteratee, accumulator) : transform(object, iteratee);
};
const results = lodashStable.map(accumulators, mapper);
let actual = lodashStable.map(results, (result) => result instanceof Foo);
assert.deepStrictEqual(actual, expected);
expected = lodashStable.map(accumulators, lodashStable.constant({ a: 1, b: 4, c: 9 }));
actual = lodashStable.map(results, lodashStable.toPlainObject);
assert.deepStrictEqual(actual, expected);
object = { a: 1, b: 2, c: 3 };
actual = lodashStable.map(accumulators, mapper);
assert.deepStrictEqual(actual, expected);
object = [1, 2, 3];
expected = lodashStable.map(accumulators, lodashStable.constant([1, 4, 9]));
actual = lodashStable.map(accumulators, mapper);
assert.deepStrictEqual(actual, expected);
});
it('should create regular arrays from typed arrays', () => {
const expected = lodashStable.map(typedArrays, stubTrue);
const actual = lodashStable.map(typedArrays, (type) => {
const Ctor = root[type],
array = Ctor ? new Ctor(new ArrayBuffer(24)) : [];
return lodashStable.isArray(transform(array, noop));
});
assert.deepStrictEqual(actual, expected);
});
it('should support an `accumulator` value', () => {
var values = [new Foo(), [1, 2, 3], { a: 1, b: 2, c: 3 }],
expected = lodashStable.map(values, lodashStable.constant([1, 4, 9]));
let actual = lodashStable.map(values, (value) =>
transform(
value,
(result, value) => {
result.push(square(value));
},
[],
),
);
assert.deepStrictEqual(actual, expected);
var object = { a: 1, b: 4, c: 9 },
expected = [object, { '0': 1, '1': 4, '2': 9 }, object];
actual = lodashStable.map(values, (value) =>
transform(
value,
(result, value, key) => {
result[key] = square(value);
},
{},
),
);
assert.deepStrictEqual(actual, expected);
lodashStable.each([[], {}], (accumulator) => {
const actual = lodashStable.map(values, (value) => transform(value, noop, accumulator));
assert.ok(lodashStable.every(actual, (result) => result === accumulator));
assert.strictEqual(transform(null, null, accumulator), accumulator);
});
});
it('should treat sparse arrays as dense', () => {
const actual = transform(Array(1), (result, value, index) => {
result[index] = String(value);
});
assert.deepStrictEqual(actual, ['undefined']);
});
it('should work without an `iteratee`', () => {
assert.ok(transform(new Foo()) instanceof Foo);
});
it('should ensure `object` is an object before using its `[[Prototype]]`', () => {
let Ctors = [Boolean, Boolean, Number, Number, Number, String, String],
values = [false, true, 0, 1, NaN, '', 'a'],
expected = lodashStable.map(values, stubObject);
const results = lodashStable.map(values, (value) => transform(value));
assert.deepStrictEqual(results, expected);
expected = lodashStable.map(values, stubFalse);
const actual = lodashStable.map(results, (value, index) => value instanceof Ctors[index]);
assert.deepStrictEqual(actual, expected);
});
it('should ensure `object` constructor is a function before using its `[[Prototype]]`', () => {
Foo.prototype.constructor = null;
assert.ok(!(transform(new Foo()) instanceof Foo));
Foo.prototype.constructor = Foo;
});
it('should create an empty object when given a falsey `object`', () => {
const expected = lodashStable.map(falsey, stubObject);
const actual = lodashStable.map(falsey, (object, index) =>
index ? transform(object) : transform(),
);
assert.deepStrictEqual(actual, expected);
});
lodashStable.each(
{
array: [1, 2, 3],
object: { a: 1, b: 2, c: 3 },
},
(object, key) => {
it(`should provide correct \`iteratee\` arguments when transforming an ${key}`, () => {
let args;
transform(object, function () {
args || (args = slice.call(arguments));
});
const first = args[0];
if (key === 'array') {
assert.ok(first !== object && lodashStable.isArray(first));
assert.deepStrictEqual(args, [first, 1, 0, object]);
} else {
assert.ok(first !== object && lodashStable.isPlainObject(first));
assert.deepStrictEqual(args, [first, 1, 'a', object]);
}
});
},
);
it('should create an object from the same realm as `object`', () => {
const objects = lodashStable.filter(
realm,
(value) => lodashStable.isObject(value) && !lodashStable.isElement(value),
);
const expected = lodashStable.map(objects, stubTrue);
const actual = lodashStable.map(objects, (object) => {
const Ctor = object.constructor,
result = transform(object);
if (result === object) {
return false;
}
if (lodashStable.isTypedArray(object)) {
return result instanceof Array;
}
return result instanceof Ctor || !(new Ctor() instanceof Ctor);
});
assert.deepStrictEqual(actual, expected);
});
});