import watch from '../../../services/note/watch'; import { publishNoteStream } from '../../stream'; import { User } from '../../../models/entities/user'; import { Note } from '../../../models/entities/note'; import { PollVotes, Users, NoteWatchings, Polls, UserProfiles } from '../../../models'; import { Not } from 'typeorm'; import { genId } from '../../../misc/gen-id'; import { createNotification } from '../../create-notification'; export default (user: User, note: Note, choice: number) => new Promise(async (res, rej) => { const poll = await Polls.findOne({ noteId: note.id }); // Check whether is valid choice if (poll.choices[choice] == null) return rej('invalid choice param'); // if already voted const exist = await PollVotes.find({ noteId: note.id, userId: user.id }); if (poll.multiple) { if (exist.some(x => x.choice === choice)) { return rej('already voted'); } } else if (exist.length !== 0) { return rej('already voted'); } // Create vote await PollVotes.save({ id: genId(), createdAt: new Date(), noteId: note.id, userId: user.id, choice: choice }); res(); // Increment votes count const index = choice + 1; // In SQL, array index is 1 based await Polls.query(`UPDATE poll SET votes[${index}] = votes[${index}] + 1 WHERE noteId = '${poll.noteId}'`); publishNoteStream(note.id, 'pollVoted', { choice: choice, userId: user.id }); // Notify createNotification(note.userId, user.id, 'pollVote', { noteId: note.id, choice: choice }); // Fetch watchers NoteWatchings.find({ noteId: note.id, userId: Not(user.id), }) .then(watchers => { for (const watcher of watchers) { createNotification(watcher.userId, user.id, 'pollVote', { noteId: note.id, choice: choice }); } }); const profile = await UserProfiles.findOne({ userId: user.id }); // ローカルユーザーが投票した場合この投稿をWatchする if (Users.isLocalUser(user) && profile.autoWatch) { watch(user.id, note); } });