// Script for lining up users and contracts/comments to make sure the denormalized avatar URLs in the contracts and // comments match the user avatar URLs. import * as admin from 'firebase-admin' import { initAdmin } from './script-init' import { DocumentCorrespondence, findDiffs, describeDiff, applyDiff, } from './denormalize' import { DocumentSnapshot, Transaction } from 'firebase-admin/firestore' initAdmin() const firestore = admin.firestore() async function getUsersById(transaction: Transaction) { const results = new Map() const users = await transaction.get(firestore.collection('users')) users.forEach((doc) => { results.set(doc.get('id'), doc) }) console.log(`Found ${results.size} unique users.`) return results } async function getContractsByUserId(transaction: Transaction) { let n = 0 const results = new Map() const contracts = await transaction.get(firestore.collection('contracts')) contracts.forEach((doc) => { const creatorId = doc.get('creatorId') const creatorContracts = results.get(creatorId) || [] creatorContracts.push(doc) results.set(creatorId, creatorContracts) n++ }) console.log(`Found ${n} contracts from ${results.size} unique users.`) return results } async function getCommentsByUserId(transaction: Transaction) { let n = 0 const results = new Map() const comments = await transaction.get(firestore.collectionGroup('comments')) comments.forEach((doc) => { const userId = doc.get('userId') const userComments = results.get(userId) || [] userComments.push(doc) results.set(userId, userComments) n++ }) console.log(`Found ${n} comments from ${results.size} unique users.`) return results } async function getAnswersByUserId(transaction: Transaction) { let n = 0 const results = new Map() const answers = await transaction.get(firestore.collectionGroup('answers')) answers.forEach((doc) => { const userId = doc.get('userId') const userAnswers = results.get(userId) || [] userAnswers.push(doc) results.set(userId, userAnswers) n++ }) console.log(`Found ${n} answers from ${results.size} unique users.`) return results } if (require.main === module) { admin.firestore().runTransaction(async (transaction) => { const [usersById, contractsByUserId, commentsByUserId, answersByUserId] = await Promise.all([ getUsersById(transaction), getContractsByUserId(transaction), getCommentsByUserId(transaction), getAnswersByUserId(transaction), ]) const usersContracts = Array.from( usersById.entries(), ([id, doc]): DocumentCorrespondence => { return [doc, contractsByUserId.get(id) || []] } ) const contractDiffs = findDiffs( usersContracts, 'avatarUrl', 'creatorAvatarUrl' ) console.log(`Found ${contractDiffs.length} contracts with mismatches.`) contractDiffs.forEach((d) => { console.log(describeDiff(d)) applyDiff(transaction, d) }) const usersComments = Array.from( usersById.entries(), ([id, doc]): DocumentCorrespondence => { return [doc, commentsByUserId.get(id) || []] } ) const commentDiffs = findDiffs(usersComments, 'avatarUrl', 'userAvatarUrl') console.log(`Found ${commentDiffs.length} comments with mismatches.`) commentDiffs.forEach((d) => { console.log(describeDiff(d)) applyDiff(transaction, d) }) const usersAnswers = Array.from( usersById.entries(), ([id, doc]): DocumentCorrespondence => { return [doc, answersByUserId.get(id) || []] } ) const answerDiffs = findDiffs(usersAnswers, 'avatarUrl', 'avatarUrl') console.log(`Found ${answerDiffs.length} answers with mismatches.`) answerDiffs.forEach((d) => { console.log(describeDiff(d)) applyDiff(transaction, d) }) }) }