2022-04-25 09:18:17 +00:00
|
|
|
const Settings = require('@overleaf/settings')
|
|
|
|
const { waitForDb } = require('../app/src/infrastructure/mongodb')
|
|
|
|
const { promiseMapWithLimit } = require('../app/src/util/promises')
|
|
|
|
const { getHardDeletedProjectIds } = require('./delete_orphaned_data_helper')
|
|
|
|
const TpdsUpdateSender = require('../app/src/Features/ThirdPartyDataStore/TpdsUpdateSender')
|
|
|
|
const { promisify } = require('util')
|
|
|
|
const { ObjectId } = require('mongodb')
|
|
|
|
const request = require('request-promise-native')
|
|
|
|
const sleep = promisify(setTimeout)
|
|
|
|
|
|
|
|
const START_OFFSET = process.env.START_OFFSET
|
|
|
|
|
|
|
|
const BATCH_SIZE = parseInt(process.env.BATCH_SIZE, 10) || 1000
|
|
|
|
const DRY_RUN = process.env.DRY_RUN !== 'false'
|
|
|
|
const READ_CONCURRENCY_SECONDARY =
|
|
|
|
parseInt(process.env.READ_CONCURRENCY_SECONDARY, 10) || 1000
|
|
|
|
const READ_CONCURRENCY_PRIMARY =
|
|
|
|
parseInt(process.env.READ_CONCURRENCY_PRIMARY, 10) || 500
|
|
|
|
const WRITE_CONCURRENCY = parseInt(process.env.WRITE_CONCURRENCY, 10) || 10
|
|
|
|
|
|
|
|
const LET_USER_DOUBLE_CHECK_INPUTS_FOR =
|
|
|
|
parseInt(process.env.LET_USER_DOUBLE_CHECK_INPUTS_FOR, 10) || 10 * 1000
|
|
|
|
|
|
|
|
async function main() {
|
|
|
|
await letUserDoubleCheckInputs()
|
|
|
|
await waitForDb()
|
|
|
|
|
|
|
|
let processed = 0
|
|
|
|
let hardDeleted = 0
|
|
|
|
let pageToken = ''
|
|
|
|
let startOffset = START_OFFSET
|
|
|
|
while (pageToken !== undefined) {
|
2022-04-26 15:19:18 +00:00
|
|
|
const { nextPageToken, entries } = await request({
|
2022-04-25 09:18:17 +00:00
|
|
|
url: `${Settings.apis.project_archiver.url}/project/list`,
|
|
|
|
json: true,
|
|
|
|
qs: {
|
|
|
|
pageToken,
|
|
|
|
startOffset,
|
|
|
|
},
|
|
|
|
})
|
|
|
|
pageToken = nextPageToken
|
|
|
|
startOffset = undefined
|
|
|
|
|
2022-04-26 15:19:18 +00:00
|
|
|
hardDeleted += await processBatch(entries)
|
|
|
|
processed += entries.length
|
2022-04-25 09:18:17 +00:00
|
|
|
console.log(
|
|
|
|
'processed:',
|
|
|
|
processed.toString().padStart(10, '0'),
|
|
|
|
'hard deleted:',
|
|
|
|
hardDeleted.toString().padStart(10, '0'),
|
|
|
|
'nextPageToken:',
|
|
|
|
nextPageToken,
|
|
|
|
'START_OFFSET:',
|
2022-04-26 15:19:18 +00:00
|
|
|
entries.pop()?.prefix
|
2022-04-25 09:18:17 +00:00
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
2022-04-26 15:19:18 +00:00
|
|
|
async function processBatch(entries) {
|
|
|
|
const projectIdToPrefix = new Map()
|
|
|
|
for (const { prefix, projectId } of entries) {
|
|
|
|
const prefixes = projectIdToPrefix.get(projectId) || []
|
|
|
|
prefixes.push(prefix)
|
|
|
|
projectIdToPrefix.set(projectId, prefixes)
|
|
|
|
}
|
|
|
|
const projectIds = Array.from(projectIdToPrefix.keys()).map(id =>
|
|
|
|
ObjectId(id)
|
|
|
|
)
|
2022-04-25 09:18:17 +00:00
|
|
|
const projectsWithOrphanedArchive = await getHardDeletedProjectIds({
|
|
|
|
projectIds,
|
|
|
|
READ_CONCURRENCY_PRIMARY,
|
|
|
|
READ_CONCURRENCY_SECONDARY,
|
|
|
|
})
|
|
|
|
|
|
|
|
await promiseMapWithLimit(
|
|
|
|
WRITE_CONCURRENCY,
|
2022-04-26 15:19:18 +00:00
|
|
|
projectsWithOrphanedArchive.flatMap(id =>
|
|
|
|
projectIdToPrefix.get(id.toString())
|
|
|
|
),
|
2022-04-25 09:18:17 +00:00
|
|
|
hardDeleteProjectArchiverData
|
|
|
|
)
|
|
|
|
return projectsWithOrphanedArchive.length
|
|
|
|
}
|
|
|
|
|
2022-04-26 15:19:18 +00:00
|
|
|
async function hardDeleteProjectArchiverData(prefix) {
|
|
|
|
console.log(`Destroying hard deleted project archive at '${prefix}/'`)
|
2022-04-25 09:18:17 +00:00
|
|
|
if (DRY_RUN) return
|
|
|
|
|
2022-04-28 10:18:26 +00:00
|
|
|
for (let i = 0; i < 10; i++) {
|
|
|
|
await sleep(1000 * i)
|
|
|
|
try {
|
|
|
|
const ok = await TpdsUpdateSender.promises.deleteProject({
|
2022-08-30 11:45:59 +00:00
|
|
|
projectId: encodeURIComponent(prefix),
|
2022-04-28 10:18:26 +00:00
|
|
|
})
|
|
|
|
if (ok) {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
} catch (e) {
|
|
|
|
console.error(`deletion failed for '${prefix}/'`, e)
|
|
|
|
}
|
2022-04-25 09:18:17 +00:00
|
|
|
}
|
2022-04-28 10:18:26 +00:00
|
|
|
throw new Error(`deletion failed for '${prefix}/', check logs`)
|
2022-04-25 09:18:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
async function letUserDoubleCheckInputs() {
|
|
|
|
console.error(
|
|
|
|
'Options:',
|
|
|
|
JSON.stringify(
|
|
|
|
{
|
|
|
|
BATCH_SIZE,
|
|
|
|
DRY_RUN,
|
|
|
|
LET_USER_DOUBLE_CHECK_INPUTS_FOR,
|
|
|
|
READ_CONCURRENCY_SECONDARY,
|
|
|
|
READ_CONCURRENCY_PRIMARY,
|
|
|
|
START_OFFSET,
|
|
|
|
WRITE_CONCURRENCY,
|
|
|
|
},
|
|
|
|
null,
|
|
|
|
2
|
|
|
|
)
|
|
|
|
)
|
|
|
|
console.error(
|
|
|
|
'Waiting for you to double check inputs for',
|
|
|
|
LET_USER_DOUBLE_CHECK_INPUTS_FOR,
|
|
|
|
'ms'
|
|
|
|
)
|
|
|
|
await sleep(LET_USER_DOUBLE_CHECK_INPUTS_FOR)
|
|
|
|
}
|
|
|
|
|
|
|
|
main()
|
|
|
|
.then(() => {
|
|
|
|
console.log('Done.')
|
|
|
|
process.exit(0)
|
|
|
|
})
|
|
|
|
.catch(error => {
|
|
|
|
console.error({ error })
|
|
|
|
process.exit(1)
|
|
|
|
})
|