-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathmiddleware.js
75 lines (62 loc) · 2 KB
/
middleware.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
import Promise from 'bluebird';
import forOwn from 'lodash.forown';
import { applyMiddleware } from 'redux';
import { REDUX_PREFETCH_INIT, REDUX_PREFETCH_SUCCESS, REDUX_PREFETCH_ERROR } from './constants.js';//
const FILTER_ACTIONS = [REDUX_PREFETCH_INIT, REDUX_PREFETCH_SUCCESS, REDUX_PREFETCH_ERROR];
class AsyncResolver {
constructor() {
this.pendingActions = [];
this.store = null;
}
resolve = (components, params) => {
const { store } = this;
return Promise
.filter(components, component => component && typeof component.fetch === 'function')
.map(component => component.fetch(store, params))
.then(this.iteratePendingActions);
}
iteratePendingActions = () => {
const { store, pendingActions } = this;
const queue = [].concat(pendingActions);
pendingActions.splice(0, pendingActions.length);
return Promise.join(...queue)
.reflect()
.then(() => {
if (pendingActions.length > 0) {
return this.iteratePendingActions();
}
})
.then(store.getState);
}
middleware() {
const { pendingActions } = this;
return store => next => action => { // eslint-line-ignore
const resultingAction = next(action);
if (FILTER_ACTIONS.indexOf(action.type) >= 0) {
return resultingAction;
}
const promises = [];
forOwn(resultingAction.payload, prop => {
if (prop && typeof prop === 'object' && typeof prop.then === 'function') {
promises.push(prop);
}
});
if (promises.length > 0) {
pendingActions.push(...promises);
return Promise.join(...promises);
}
return resultingAction;
};
}
}
export default function initResolver(next) {
return (reducer, initialState) => {
const asyncResolver = new AsyncResolver();
const store = applyMiddleware(asyncResolver.middleware())(next)(reducer, initialState);
asyncResolver.store = store;
store.resolve = asyncResolver.resolve;
return {
...store,
};
};
}