knex/lib/util/batchInsert.js

60 lines
1.3 KiB
JavaScript
Raw Normal View History

const chunk = require('lodash/chunk');
const flatten = require('lodash/flatten');
const isNumber = require('lodash/isNumber');
2019-10-11 11:12:56 +03:00
const delay = require('./delay');
module.exports = function batchInsert(
client,
tableName,
batch,
chunkSize = 1000
) {
let returning = void 0;
let transaction = null;
2019-10-11 11:12:56 +03:00
const runInTransaction = (cb) => {
if (transaction) {
return cb(transaction);
}
return client.transaction(cb);
};
2019-10-11 11:12:56 +03:00
return Object.assign(
Promise.resolve().then(async () => {
if (!isNumber(chunkSize) || chunkSize < 1) {
2019-10-11 11:12:56 +03:00
throw new TypeError(`Invalid chunkSize: ${chunkSize}`);
}
if (!Array.isArray(batch)) {
2019-10-11 11:12:56 +03:00
throw new TypeError(
`Invalid batch: Expected array, got ${typeof batch}`
);
}
2019-10-11 11:12:56 +03:00
const chunks = chunk(batch, chunkSize);
2019-10-11 11:12:56 +03:00
//Next tick to ensure wrapper functions are called if needed
await delay(1);
return runInTransaction(async (tr) => {
const chunksResults = [];
for (const items of chunks) {
chunksResults.push(await tr(tableName).insert(items, returning));
}
return flatten(chunksResults);
});
}),
{
returning(columns) {
returning = columns;
return this;
},
transacting(tr) {
transaction = tr;
return this;
},
}
);
};