da83322200
* wrap follow/unfollow and block/unblock as job queue * create import job to follow in each iteration * make relationship jobs concurrent * replace to job queue if called repeatedly * use addBulk to import * omit stream when importing * fix job caller * use ThinUser instead of User to reduce redis memory consumption * createImportFollowingToDbJobの呼び出し方を変える, 型補強 * Force ThinUser * オブジェクト操作のみのメソッド名はgenerate...Data * Force ThinUser in generateRelationshipJobData * silent bulk unfollow at admin api endpoint --------- Co-authored-by: tamaina <tamaina@hotmail.co.jp>
92 lines
2.5 KiB
TypeScript
92 lines
2.5 KiB
TypeScript
import { IsNull, Not } from 'typeorm';
|
|
import { Inject, Injectable } from '@nestjs/common';
|
|
import { Endpoint } from '@/server/api/endpoint-base.js';
|
|
import type { UsersRepository, FollowingsRepository } from '@/models/index.js';
|
|
import type { User } from '@/models/entities/User.js';
|
|
import type { RelationshipJobData } from '@/queue/types.js';
|
|
import { ModerationLogService } from '@/core/ModerationLogService.js';
|
|
import { UserSuspendService } from '@/core/UserSuspendService.js';
|
|
import { DI } from '@/di-symbols.js';
|
|
import { bindThis } from '@/decorators.js';
|
|
import { RoleService } from '@/core/RoleService.js';
|
|
import { QueueService } from '@/core/QueueService.js';
|
|
|
|
export const meta = {
|
|
tags: ['admin'],
|
|
|
|
requireCredential: true,
|
|
requireModerator: true,
|
|
} as const;
|
|
|
|
export const paramDef = {
|
|
type: 'object',
|
|
properties: {
|
|
userId: { type: 'string', format: 'misskey:id' },
|
|
},
|
|
required: ['userId'],
|
|
} as const;
|
|
|
|
// eslint-disable-next-line import/no-default-export
|
|
@Injectable()
|
|
export default class extends Endpoint<typeof meta, typeof paramDef> {
|
|
constructor(
|
|
@Inject(DI.usersRepository)
|
|
private usersRepository: UsersRepository,
|
|
|
|
@Inject(DI.followingsRepository)
|
|
private followingsRepository: FollowingsRepository,
|
|
|
|
private userSuspendService: UserSuspendService,
|
|
private roleService: RoleService,
|
|
private moderationLogService: ModerationLogService,
|
|
private queueService: QueueService,
|
|
) {
|
|
super(meta, paramDef, async (ps, me) => {
|
|
const user = await this.usersRepository.findOneBy({ id: ps.userId });
|
|
|
|
if (user == null) {
|
|
throw new Error('user not found');
|
|
}
|
|
|
|
if (await this.roleService.isModerator(user)) {
|
|
throw new Error('cannot suspend moderator account');
|
|
}
|
|
|
|
await this.usersRepository.update(user.id, {
|
|
isSuspended: true,
|
|
});
|
|
|
|
this.moderationLogService.insertModerationLog(me, 'suspend', {
|
|
targetId: user.id,
|
|
});
|
|
|
|
(async () => {
|
|
await this.userSuspendService.doPostSuspend(user).catch(e => {});
|
|
await this.unFollowAll(user).catch(e => {});
|
|
})();
|
|
});
|
|
}
|
|
|
|
@bindThis
|
|
private async unFollowAll(follower: User) {
|
|
const followings = await this.followingsRepository.find({
|
|
where: {
|
|
followerId: follower.id,
|
|
followeeId: Not(IsNull()),
|
|
},
|
|
});
|
|
|
|
const jobs: RelationshipJobData[] = [];
|
|
for (const following of followings) {
|
|
if (following.followeeId && following.followerId) {
|
|
jobs.push({
|
|
from: { id: following.followerId },
|
|
to: { id: following.followeeId },
|
|
silent: true,
|
|
});
|
|
}
|
|
}
|
|
this.queueService.createUnfollowJob(jobs);
|
|
}
|
|
}
|