* Align false conditions before true * Update documents.delete endpoint for permanent delete * Add permanent delete to events table and integrate with socket.io * Add permanent delete to document menu * Update parentDocumentId of direct child to null * Add translation * Add test for permanent delete * Add space * Update app/scenes/DocumentPermanentDelete.js Co-authored-by: Tom Moor <tom.moor@gmail.com> * Update app/stores/DocumentsStore.js Co-authored-by: Tom Moor <tom.moor@gmail.com> * Update server/commands/documentPermanentDeleter.js Co-authored-by: Tom Moor <tom.moor@gmail.com> * Update app/scenes/DocumentPermanentDelete.js Co-authored-by: Tom Moor <tom.moor@gmail.com> * Change socket room from team to collection * Add translation * Create log func for commands * Move tests from utils to permanentDeleter command * Add additional tests * Set redirect to trash * Return promise from beforeEach * Add undeleted documents validation * Include deleteAt attribute in db query * Update server/commands/documentPermanentDeleter.js Co-authored-by: Tom Moor <tom.moor@gmail.com> * tweak language Co-authored-by: Tom Moor <tom.moor@gmail.com>
65 lines
1.6 KiB
JavaScript
65 lines
1.6 KiB
JavaScript
// @flow
|
|
import debug from "debug";
|
|
import { Document, Attachment } from "../models";
|
|
import { sequelize } from "../sequelize";
|
|
import parseAttachmentIds from "../utils/parseAttachmentIds";
|
|
|
|
const log = debug("commands");
|
|
|
|
export async function documentPermanentDeleter(documents: Document[]) {
|
|
const activeDocument = documents.find((doc) => !doc.deletedAt);
|
|
|
|
if (activeDocument) {
|
|
throw new Error(
|
|
`Cannot permanently delete ${activeDocument.id} document. Please delete it and try again.`
|
|
);
|
|
}
|
|
|
|
const query = `
|
|
SELECT COUNT(id)
|
|
FROM documents
|
|
WHERE "searchVector" @@ to_tsquery('english', :query) AND
|
|
"teamId" = :teamId AND
|
|
"id" != :documentId
|
|
`;
|
|
|
|
for (const document of documents) {
|
|
const attachmentIds = parseAttachmentIds(document.text);
|
|
|
|
for (const attachmentId of attachmentIds) {
|
|
const [{ count }] = await sequelize.query(query, {
|
|
type: sequelize.QueryTypes.SELECT,
|
|
replacements: {
|
|
documentId: document.id,
|
|
teamId: document.teamId,
|
|
query: attachmentId,
|
|
},
|
|
});
|
|
|
|
if (parseInt(count) === 0) {
|
|
const attachment = await Attachment.findOne({
|
|
where: {
|
|
teamId: document.teamId,
|
|
id: attachmentId,
|
|
},
|
|
});
|
|
|
|
if (attachment) {
|
|
await attachment.destroy();
|
|
|
|
log(`Attachment ${attachmentId} deleted`);
|
|
} else {
|
|
log(`Unknown attachment ${attachmentId} ignored`);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return Document.scope("withUnpublished").destroy({
|
|
where: {
|
|
id: documents.map((document) => document.id),
|
|
},
|
|
force: true,
|
|
});
|
|
}
|