yumechi-no-kuni/packages/backend/src/core/PollService.ts

104 lines
3.4 KiB
TypeScript
Raw Normal View History

2022-09-17 13:27:08 -05:00
import { Inject, Injectable } from '@nestjs/common';
import { DI } from '@/di-symbols.js';
2023-02-13 00:28:07 -06:00
import type { NotesRepository, UsersRepository, PollsRepository, PollVotesRepository, User } from '@/models/index.js';
2022-09-17 13:27:08 -05:00
import type { Note } from '@/models/entities/Note.js';
import { RelayService } from '@/core/RelayService.js';
import { IdService } from '@/core/IdService.js';
import { GlobalEventService } from '@/core/GlobalEventService.js';
2022-12-03 19:16:03 -06:00
import { ApRendererService } from '@/core/activitypub/ApRendererService.js';
import { UserEntityService } from '@/core/entities/UserEntityService.js';
import { ApDeliverManagerService } from '@/core/activitypub/ApDeliverManagerService.js';
import { bindThis } from '@/decorators.js';
2023-02-03 21:40:40 -06:00
import { UserBlockingService } from '@/core/UserBlockingService.js';
2022-09-17 13:27:08 -05:00
@Injectable()
export class PollService {
constructor(
@Inject(DI.usersRepository)
private usersRepository: UsersRepository,
@Inject(DI.notesRepository)
private notesRepository: NotesRepository,
@Inject(DI.pollsRepository)
private pollsRepository: PollsRepository,
@Inject(DI.pollVotesRepository)
private pollVotesRepository: PollVotesRepository,
private userEntityService: UserEntityService,
private idService: IdService,
private relayService: RelayService,
2023-02-03 19:02:03 -06:00
private globalEventService: GlobalEventService,
2023-02-03 21:40:40 -06:00
private userBlockingService: UserBlockingService,
2022-09-17 13:27:08 -05:00
private apRendererService: ApRendererService,
private apDeliverManagerService: ApDeliverManagerService,
) {
}
@bindThis
2023-02-13 00:28:07 -06:00
public async vote(user: User, note: Note, choice: number) {
2022-09-17 13:27:08 -05:00
const poll = await this.pollsRepository.findOneBy({ noteId: note.id });
if (poll == null) throw new Error('poll not found');
// Check whether is valid choice
if (poll.choices[choice] == null) throw new Error('invalid choice param');
// Check blocking
if (note.userId !== user.id) {
2023-02-03 21:40:40 -06:00
const blocked = await this.userBlockingService.checkBlocked(note.userId, user.id);
if (blocked) {
2022-09-17 13:27:08 -05:00
throw new Error('blocked');
}
}
// if already voted
const exist = await this.pollVotesRepository.findBy({
noteId: note.id,
userId: user.id,
});
if (poll.multiple) {
if (exist.some(x => x.choice === choice)) {
throw new Error('already voted');
}
} else if (exist.length !== 0) {
throw new Error('already voted');
}
// Create vote
await this.pollVotesRepository.insert({
id: this.idService.genId(),
createdAt: new Date(),
noteId: note.id,
userId: user.id,
choice: choice,
});
// Increment votes count
const index = choice + 1; // In SQL, array index is 1 based
await this.pollsRepository.query(`UPDATE poll SET votes[${index}] = votes[${index}] + 1 WHERE "noteId" = '${poll.noteId}'`);
2023-02-03 19:02:03 -06:00
this.globalEventService.publishNoteStream(note.id, 'pollVoted', {
2022-09-17 13:27:08 -05:00
choice: choice,
userId: user.id,
});
}
@bindThis
2022-09-17 13:27:08 -05:00
public async deliverQuestionUpdate(noteId: Note['id']) {
const note = await this.notesRepository.findOneBy({ id: noteId });
if (note == null) throw new Error('note not found');
const user = await this.usersRepository.findOneBy({ id: note.userId });
if (user == null) throw new Error('note not found');
if (this.userEntityService.isLocalUser(user)) {
2023-02-12 03:47:30 -06:00
const content = this.apRendererService.addContext(this.apRendererService.renderUpdate(await this.apRendererService.renderNote(note, false), user));
2022-09-17 13:27:08 -05:00
this.apDeliverManagerService.deliverToFollowers(user, content);
this.relayService.deliverToRelays(user, content);
}
}
}