overleaf/services/web/scripts/delete_dangling_comments.mjs
Eric Mc Sween 5f6de9caf0 Merge pull request #23706 from overleaf/em-fix-dangling-comments-script
Fix dangling comments script when doc does not have ranges

GitOrigin-RevId: 31cf25ecf79f7ab0ed872c403d31b2c5064b4c76
2025-02-20 09:06:05 +00:00

85 lines
2.5 KiB
JavaScript

// @ts-check
import minimist from 'minimist'
import ChatApiHandler from '../app/src/Features/Chat/ChatApiHandler.js'
import DocumentUpdaterHandler from '../app/src/Features/DocumentUpdater/DocumentUpdaterHandler.js'
import DocstoreManager from '../app/src/Features/Docstore/DocstoreManager.js'
import HistoryManager from '../app/src/Features/History/HistoryManager.js'
import { db, ObjectId } from '../app/src/infrastructure/mongodb.js'
const OPTS = parseArgs()
function usage() {
console.error(
'Usage: node delete_dangling_comments.mjs [--commit] PROJECT_ID...'
)
}
function parseArgs() {
const args = minimist(process.argv.slice(2), {
boolean: ['commit'],
})
if (args._.length === 0) {
usage()
process.exit(0)
}
return {
projectIds: args._,
commit: args.commit,
}
}
async function processProject(projectId) {
console.log(`Processing project ${projectId}...`)
await DocumentUpdaterHandler.promises.flushProjectToMongoAndDelete(projectId)
const docRanges = await DocstoreManager.promises.getAllRanges(projectId)
const threads = await ChatApiHandler.promises.getThreads(projectId)
const threadIds = new Set(Object.keys(threads))
let commentsDeleted = 0
for (const doc of docRanges) {
const commentsDeletedInDoc = await processDoc(projectId, doc, threadIds)
commentsDeleted += commentsDeletedInDoc
}
if (OPTS.commit) {
console.log(`${commentsDeleted} comments deleted`)
if (commentsDeleted > 0) {
console.log(`Resyncing history for project ${projectId}`)
await HistoryManager.promises.resyncProject(projectId)
}
}
}
async function processDoc(projectId, doc, threadIds) {
let commentsDeleted = 0
for (const comment of doc.ranges?.comments ?? []) {
const threadId = comment.op.t
if (!threadIds.has(threadId)) {
if (OPTS.commit) {
console.log(`Deleting dangling comment ${comment.op.t}...`)
await deleteComment(doc._id, threadId)
commentsDeleted += 1
} else {
console.log(`Would delete dangling comment ${comment.op.t}...`)
}
}
}
return commentsDeleted
}
async function deleteComment(docId, threadId) {
await db.docs.updateOne(
{ _id: new ObjectId(docId) },
{
$pull: { 'ranges.comments': { 'op.t': new ObjectId(threadId) } },
}
)
}
// Main loop
for (const projectId of OPTS.projectIds) {
await processProject(projectId)
}
if (!OPTS.commit) {
console.log('This was a dry run. Rerun with --commit to apply changes')
}
process.exit(0)