From 35c75bbebf2bc8e55d9027ad4e84234b70dd0c6b Mon Sep 17 00:00:00 2001 From: Laura Hausmann Date: Sat, 4 Nov 2023 23:16:26 +0100 Subject: [PATCH] [backend] Reset poll votes when choices change on note edit --- .../src/remote/activitypub/models/note.ts | 57 +++++++++------- packages/backend/src/services/note/edit.ts | 66 +++++++++++-------- 2 files changed, 73 insertions(+), 50 deletions(-) diff --git a/packages/backend/src/remote/activitypub/models/note.ts b/packages/backend/src/remote/activitypub/models/note.ts index 39c17cc92..7fe1a15e1 100644 --- a/packages/backend/src/remote/activitypub/models/note.ts +++ b/packages/backend/src/remote/activitypub/models/note.ts @@ -25,6 +25,7 @@ import { Notes, NoteEdits, DriveFiles, + PollVotes, } from "@/models/index.js"; import type { IMentionedRemoteUsers, Note } from "@/models/entities/note.js"; import type { IObject, IPost } from "../type.js"; @@ -710,30 +711,40 @@ export async function updateNote(value: string | IObject, resolver?: Resolver) { userHost: actor.host, }); updating = true; - } else if ( - dbPoll.multiple !== poll.multiple || - dbPoll.expiresAt !== poll.expiresAt || - dbPoll.noteVisibility !== note.visibility || - JSON.stringify(dbPoll.choices) !== JSON.stringify(poll.choices) - ) { - await Polls.update( - { noteId: note.id }, - { - choices: poll?.choices, - multiple: poll?.multiple, - votes: poll?.votes, - expiresAt: poll?.expiresAt, - noteVisibility: - note.visibility === "hidden" ? "home" : note.visibility, - }, - ); - updating = true; } else { - for (let i = 0; i < poll.choices.length; i++) { - if (dbPoll.votes[i] !== poll.votes?.[i]) { - await Polls.update({ noteId: note.id }, { votes: poll?.votes }); - updating = true; - break; + const choicesChanged = JSON.stringify(dbPoll.choices) !== JSON.stringify(poll.choices); + + if ( + dbPoll.multiple !== poll.multiple || + dbPoll.expiresAt !== poll.expiresAt || + dbPoll.noteVisibility !== note.visibility || + choicesChanged + ) { + await Polls.update( + { noteId: note.id }, + { + choices: poll?.choices, + multiple: poll?.multiple, + votes: poll?.votes, + expiresAt: poll?.expiresAt, + noteVisibility: + note.visibility === "hidden" ? "home" : note.visibility, + }, + ); + + // Reset votes + if (choicesChanged) { + await PollVotes.delete({ noteId: dbPoll.noteId }); + } + + updating = true; + } else { + for (let i = 0; i < poll.choices.length; i++) { + if (dbPoll.votes[i] !== poll.votes?.[i]) { + await Polls.update({ noteId: note.id }, { votes: poll?.votes }); + updating = true; + break; + } } } } diff --git a/packages/backend/src/services/note/edit.ts b/packages/backend/src/services/note/edit.ts index b72a9f9d3..4b8c8090c 100644 --- a/packages/backend/src/services/note/edit.ts +++ b/packages/backend/src/services/note/edit.ts @@ -10,10 +10,12 @@ import { extractHashtags } from "@/misc/extract-hashtags.js"; import type { IMentionedRemoteUsers } from "@/models/entities/note.js"; import { Note } from "@/models/entities/note.js"; import { - Users, - Notes, - UserProfiles, - Polls, NoteEdits, + Users, + Notes, + UserProfiles, + Polls, + NoteEdits, + PollVotes, } from "@/models/index.js"; import type { DriveFile } from "@/models/entities/drive-file.js"; import { In } from "typeorm"; @@ -121,30 +123,40 @@ export default async function ( userHost: user.host, }); publishing = true; - } else if ( - dbPoll.multiple !== data.poll.multiple || - dbPoll.expiresAt !== data.poll.expiresAt || - dbPoll.noteVisibility !== note.visibility || - JSON.stringify(dbPoll.choices) !== JSON.stringify(data.poll.choices) - ) { - await Polls.update( - { noteId: note.id }, - { - choices: data.poll?.choices, - multiple: data.poll?.multiple, - votes: data.poll?.votes, - expiresAt: data.poll?.expiresAt, - noteVisibility: - note.visibility === "hidden" ? "home" : note.visibility, - }, - ); - publishing = true; } else { - for (let i = 0; i < data.poll.choices.length; i++) { - if (dbPoll.votes[i] !== data.poll.votes?.[i]) { - await Polls.update({ noteId: note.id }, { votes: data.poll?.votes }); - publishing = true; - break; + const choicesChanged = JSON.stringify(dbPoll.choices) !== JSON.stringify(data.poll.choices); + + if ( + dbPoll.multiple !== data.poll.multiple || + dbPoll.expiresAt !== data.poll.expiresAt || + dbPoll.noteVisibility !== note.visibility || + choicesChanged + ) { + await Polls.update( + { noteId: note.id }, + { + choices: data.poll?.choices, + multiple: data.poll?.multiple, + votes: choicesChanged ? new Array(data.poll.choices.length).fill(0) : data.poll?.votes, + expiresAt: data.poll?.expiresAt, + noteVisibility: + note.visibility === "hidden" ? "home" : note.visibility, + }, + ); + + // Reset votes + if (JSON.stringify(dbPoll.choices) !== JSON.stringify(data.poll.choices)) { + await PollVotes.delete({ noteId: dbPoll.noteId }); + } + + publishing = true; + } else { + for (let i = 0; i < data.poll.choices.length; i++) { + if (dbPoll.votes[i] !== data.poll.votes?.[i]) { + await Polls.update({ noteId: note.id }, { votes: data.poll?.votes }); + publishing = true; + break; + } } } }