So the problem is that I have a large array of objects, let's say a 1000 that I need to send to our API in NGRX Effect and then dispatch a new Action with the response, but API takes too long to process such a large requests at once, so I want to divide this array into smaller chunks, send a request with each of them one after another and then combine responses from all of them into a single action. I've tried it with mapping this large array of data into an array of Observables with from()
operator, but couldn't figure out what to use next after it was mapped, I've tried casting it to Promises and use Promise.all() on them but again, no luck. In summary what I need to do is this:
Here is a simplified Effect that I'm using currently and what I need to divide is action.rows
:
loadRows$ = createEffect(() => {
return this.actions$.pipe(
ofType(RowsActions.LoadRows),
switchMap((action) => {
return this.eventsService.getBatchRows(action.rows).pipe(
switchMap((response) => {
return [
new RowsLoaded({ rows: response.rows }),
new LoadedTableData(response.rows),
];
}),
catchError(() => {
return of(new RowsFailedToLoad());
})
);
})
);
});
have implemented similiar logic, if one chunk will have 100 items - it still to big number.
I am running ALL items paralelly.
UPD:
if you will change mergeMap
to concatMap
- all calls will be sequential
it does not matter if it is single row or chunk of rows
from(action.rows)
.pipe(
mergeMap(id => api(id).pipe(catchError(e => EMPTY))),
toArray())
)
here is working example: https://codesandbox.io/s/rxjs-playground-forked-u8c7jd?file=/src/index.js
FINAL VERSION:
const { of, from } = require("rxjs");
const {
switchMap,
concatMap,
toArray,
catchError,
map,
tap
} = require("rxjs/operators");
const chunkArray = (arr, size) =>
arr.length > size
? [arr.slice(0, size), ...chunkArray(arr.slice(size), size)]
: [arr];
const getBatchRows = (rows) => of({ response: { rows } });
of({ rows: ["id1", "id2", "id3", "id4", "id5"] })
.pipe(
switchMap((action) =>
from(chunkArray(action.rows, 2))
.pipe(
concatMap((rows) =>
getBatchRows(rows).pipe(catchError((e) => of({})))
)
)
.pipe(toArray())
),
tap((data) => console.log(data)),
map((responseArray) => responseArray.map((i) => i.response.rows).flat()),
map(rows => [
new RowsLoaded({ rows }),
new LoadedTableData(rows),
])
)
.subscribe((data) => {
console.log(data);
});
to use chunk way - almost nothing to do
will use
const chunkArray = (arr, size) =>
arr.length > size
? [arr.slice(0, size), ...chunkArray(arr.slice(size), size)]
: [arr];
and use here:
from(chunkArray(action.rows, 100))
//and change this
mergeMap(rows => this.eventsService.getBatchRows(rows)