Skip to content

Commit c5855d2

Browse files
committed
[changed] createChainedFunction to chain many functions, and to throw if non-functions are provided.
1 parent ed7b5b6 commit c5855d2

File tree

2 files changed

+87
-12
lines changed

2 files changed

+87
-12
lines changed

src/utils/createChainedFunction.js

+16-12
Original file line numberDiff line numberDiff line change
@@ -4,22 +4,26 @@
44
* Will only create a new function if needed,
55
* otherwise will pass back existing functions or null.
66
*
7-
* @param {function} one
8-
* @param {function} two
7+
* @param {function} functions to chain
98
* @returns {function|null}
109
*/
11-
function createChainedFunction(one, two) {
12-
let hasOne = typeof one === 'function';
13-
let hasTwo = typeof two === 'function';
10+
function createChainedFunction(...funcs) {
11+
return funcs
12+
.filter(f => f != null)
13+
.reduce((acc, f) => {
14+
if (typeof f !== 'function') {
15+
throw new Error('Invalid Argument Type, must only provide functions, undefined, or null.');
16+
}
1417

15-
if (!hasOne && !hasTwo) { return null; }
16-
if (!hasOne) { return two; }
17-
if (!hasTwo) { return one; }
18+
if (acc === null) {
19+
return f;
20+
}
1821

19-
return function chainedFunction() {
20-
one.apply(this, arguments);
21-
two.apply(this, arguments);
22-
};
22+
return function chainedFunction(...args) {
23+
acc.apply(this, args);
24+
f.apply(this, args);
25+
};
26+
}, null);
2327
}
2428

2529
export default createChainedFunction;
+71
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,71 @@
1+
/* eslint no-new-func: 0 */
2+
import createChainedFunction from '../../src/utils/createChainedFunction';
3+
4+
describe('createChainedFunction', function() {
5+
it('returns null with no arguments', function() {
6+
expect(createChainedFunction()).to.equal(null);
7+
});
8+
9+
it('returns original function when single function is provided', function() {
10+
const func1 = sinon.stub();
11+
createChainedFunction(func1).should.equal(func1);
12+
});
13+
14+
it('wraps two functions with another that invokes both when called', function() {
15+
const func1 = sinon.stub();
16+
const func2 = sinon.stub();
17+
const chained = createChainedFunction(func1, func2);
18+
19+
chained
20+
.should.not.equal(func1)
21+
.and.should.not.equal(func2);
22+
23+
func1.should.not.have.been.called;
24+
func2.should.not.have.been.called;
25+
26+
chained();
27+
28+
func1.should.have.been.calledOnce;
29+
func2.should.have.been.calledOnce;
30+
});
31+
32+
it('wraps multiple functions and invokes them in the order provided', function() {
33+
const results = [];
34+
const func1 = () => results.push(1);
35+
const func2 = () => results.push(2);
36+
const func3 = () => results.push(3);
37+
const chained = createChainedFunction(func1, func2, func3);
38+
chained();
39+
results.should.eql([1, 2, 3]);
40+
});
41+
42+
it('forwards arguments to all chained functions', function() {
43+
const in1 = 'herpa derpa';
44+
const in2 = {
45+
herpa: 'derpa'
46+
};
47+
48+
const func = (arg1, arg2) => {
49+
arg1.should.equal(in1);
50+
arg2.should.equal(in2);
51+
};
52+
53+
const chained = createChainedFunction(func, func, func);
54+
chained(in1, in2);
55+
});
56+
57+
it('throws when func is not provided', function() {
58+
expect(() => {
59+
createChainedFunction({ herpa: 'derpa' });
60+
}).to.throw(/Invalid Argument Type/);
61+
});
62+
63+
it('works with new Function call', function() {
64+
const results = [];
65+
const func1 = new Function('results', 'results.push(1);');
66+
const func2 = new Function('results', 'results.push(2);');
67+
const chained = createChainedFunction(func1, func2);
68+
chained(results);
69+
results.should.eql([1, 2]);
70+
});
71+
});

0 commit comments

Comments
 (0)