From afb1f6b601dac0b25e15a20b913d6c7c5c5965df Mon Sep 17 00:00:00 2001 From: WeiJun0827 Date: Thu, 6 Oct 2022 12:39:45 +0800 Subject: [PATCH] =?UTF-8?q?=F0=9F=8E=A8=20Improve=20batch=20guard=20to=20p?= =?UTF-8?q?rivent=20race=20condition?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/util/api/likernft/user.js | 30 +++++++++++++++++++----------- 1 file changed, 19 insertions(+), 11 deletions(-) diff --git a/src/util/api/likernft/user.js b/src/util/api/likernft/user.js index 31ed8d034..68d350ce8 100644 --- a/src/util/api/likernft/user.js +++ b/src/util/api/likernft/user.js @@ -1,28 +1,36 @@ import { db } from '../../firebase'; import { getNFTISCNOwner } from '../../cosmos/nft'; +const BATCH_SIZE = 200; + export async function filterOwnedClassIds(iscnDocs, wallet) { const classIdSet = new Set(); iscnDocs.forEach((doc) => { classIdSet.add(doc.data().classId); }); - let count = 0; - let batch = db.batch(); - const promises = iscnDocs.map(async (doc) => { + const docsToUpdate = []; + const checkOwnerPromises = iscnDocs.map(async (doc) => { const iscnPrefix = decodeURIComponent(doc.id); const owner = await getNFTISCNOwner(iscnPrefix); if (owner && owner !== wallet) { + docsToUpdate.push(doc); classIdSet.delete(doc.data().classId); - batch.update({ ownerWallet: owner }); - count += 1; - if (count % 200 === 0) { - await batch.commit(); - batch = db.batch(); - } } }); - await Promise.all(promises); - if (count % 200) await batch.commit(); + await Promise.all(checkOwnerPromises); + + if (docsToUpdate.length) { + const batches = []; + for (let i = 0; i < docsToUpdate.length; i += BATCH_SIZE) { + batches.push(docsToUpdate.slice(i, i + BATCH_SIZE)); + } + const updatePromises = batches.map((docs) => { + const batch = db.batch(); + docs.forEach(doc => batch.update(doc.ref, { ownerWallet: wallet })); + return batch.commit(); + }); + await Promise.all(updatePromises); + } return Array.from(classIdSet); }