From c5cb285b30cf3187ff774d739a1fc8838fdb2f3f Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Bart=C5=82omiej=20Klocek?= Date: Mon, 30 Sep 2024 11:05:15 +0200 Subject: [PATCH] [keyserver] Remove holders in account deleters Summary: Address [[ https://linear.app/comm/issue/ENG-9353/delete-holders-when-deleting-account | ENG-9353 ]]. Before calling `DELETE FROM uploads`, we need to select the `extra` column first and remove blob holders if present in the column JSON. Depends on D13513 Test Plan: Enabled blob-hosted user avatars. Created an account, set image avatar, then deleted the account and watched the logs. Similiarly to D13513, a blob was uploaded and then its holder was removed Reviewers: tomek, ashoat Reviewed By: ashoat Differential Revision: https://phab.comm.dev/D13514 --- keyserver/src/deleters/account-deleters.js | 35 +++++++++++++++++++--- 1 file changed, 31 insertions(+), 4 deletions(-) diff --git a/keyserver/src/deleters/account-deleters.js b/keyserver/src/deleters/account-deleters.js index 6a869dd3f5..a37276d13c 100644 --- a/keyserver/src/deleters/account-deleters.js +++ b/keyserver/src/deleters/account-deleters.js @@ -17,9 +17,38 @@ import { fetchUsername, } from '../fetchers/user-fetchers.js'; import { rescindPushNotifs } from '../push/rescind.js'; +import { removeBlobHolders } from '../services/blob.js'; import { createNewAnonymousCookie } from '../session/cookies.js'; import type { Viewer, AnonymousViewerData } from '../session/viewer.js'; import { fetchOlmAccount } from '../updaters/olm-account-updater.js'; +import { blobHoldersFromUploadRows } from '../uploads/media-utils.js'; + +async function deleteUploadsForUser(deletedUserID: string): Promise { + try { + const [holderRows] = await dbQuery(SQL` + SELECT extra + FROM uploads + WHERE user_container = ${deletedUserID} + `); + const blobHolders = blobHoldersFromUploadRows(holderRows); + await removeBlobHolders(blobHolders); + await dbQuery(SQL` + DELETE u, i + FROM uploads u + LEFT JOIN ids i on i.id = u.id + WHERE u.user_container = ${deletedUserID}; + `); + } catch (err) { + // unassign uploads so the deletion will be retried + // by the `deleteUnassignedUploads()` + await dbQuery(SQL` + UPDATE uploads + SET user_container = NULL + WHERE user_container = ${deletedUserID}; + `); + throw err; + } +} async function deleteAccount(viewer: Viewer): Promise { if (!viewer.loggedIn) { @@ -33,6 +62,8 @@ async function deleteAccount(viewer: Viewer): Promise { (user: UserInfo): boolean => user.id !== deletedUserID, ); + ignorePromiseRejections(deleteUploadsForUser(deletedUserID)); + // TODO: if this results in any orphaned orgs, convert them to chats const deletionQuery = SQL` START TRANSACTION; @@ -60,10 +91,6 @@ async function deleteAccount(viewer: Viewer): Promise { FROM reports r LEFT JOIN ids i ON i.id = r.id WHERE r.user = ${deletedUserID}; - DELETE u, i - FROM uploads u - LEFT JOIN ids i on i.id = u.id - WHERE u.user_container = ${deletedUserID}; DELETE FROM relationships_undirected WHERE user1 = ${deletedUserID}; DELETE FROM relationships_undirected WHERE user2 = ${deletedUserID}; DELETE FROM relationships_directed WHERE user1 = ${deletedUserID};