import { Inject, Injectable } from '@nestjs/common'; import { IsNull, In, MoreThan, Not } from 'typeorm'; import { bindThis } from '@/decorators.js'; import { DI } from '@/di-symbols.js'; import type { Config } from '@/config.js'; import type { LocalUser } from '@/models/entities/User.js'; import type { BlockingsRepository, FollowingsRepository, InstancesRepository, Muting, MutingsRepository, UserListJoiningsRepository, UsersRepository } from '@/models/index.js'; import type { RelationshipJobData, ThinUser } from '@/queue/types.js'; import type { User } from '@/models/entities/User.js'; import { AccountUpdateService } from '@/core/AccountUpdateService.js'; import { GlobalEventService } from '@/core/GlobalEventService.js'; import { QueueService } from '@/core/QueueService.js'; import { RelayService } from '@/core/RelayService.js'; import { UserFollowingService } from '@/core/UserFollowingService.js'; import { ApDeliverManagerService } from '@/core/activitypub/ApDeliverManagerService.js'; import { ApRendererService } from '@/core/activitypub/ApRendererService.js'; import { UserEntityService } from '@/core/entities/UserEntityService.js'; import { CacheService } from '@/core/CacheService.js'; import { ProxyAccountService } from '@/core/ProxyAccountService.js'; import { FederatedInstanceService } from '@/core/FederatedInstanceService.js'; import { MetaService } from '@/core/MetaService.js'; import InstanceChart from '@/core/chart/charts/instance.js'; import PerUserFollowingChart from '@/core/chart/charts/per-user-following.js'; @Injectable() export class AccountMoveService { constructor( @Inject(DI.config) private config: Config, @Inject(DI.usersRepository) private usersRepository: UsersRepository, @Inject(DI.followingsRepository) private followingsRepository: FollowingsRepository, @Inject(DI.blockingsRepository) private blockingsRepository: BlockingsRepository, @Inject(DI.mutingsRepository) private mutingsRepository: MutingsRepository, @Inject(DI.userListJoiningsRepository) private userListJoiningsRepository: UserListJoiningsRepository, @Inject(DI.instancesRepository) private instancesRepository: InstancesRepository, private userEntityService: UserEntityService, private apRendererService: ApRendererService, private apDeliverManagerService: ApDeliverManagerService, private globalEventService: GlobalEventService, private userFollowingService: UserFollowingService, private accountUpdateService: AccountUpdateService, private proxyAccountService: ProxyAccountService, private perUserFollowingChart: PerUserFollowingChart, private federatedInstanceService: FederatedInstanceService, private instanceChart: InstanceChart, private metaService: MetaService, private relayService: RelayService, private cacheService: CacheService, private queueService: QueueService, ) { } /** * Move a local account to a new account. * * After delivering Move activity, its local followers unfollow the old account and then follow the new one. */ @bindThis public async moveFromLocal(src: LocalUser, dst: User): Promise { const dstUri = this.getUserUri(dst); // add movedToUri to indicate that the user has moved const update = {} as Partial; update.alsoKnownAs = src.alsoKnownAs?.concat([dstUri]) ?? [dstUri]; update.movedToUri = dstUri; await this.usersRepository.update(src.id, update); src = Object.assign(src, update); const srcPerson = await this.apRendererService.renderPerson(src); const updateAct = this.apRendererService.addContext(this.apRendererService.renderUpdate(srcPerson, src)); await this.apDeliverManagerService.deliverToFollowers(src, updateAct); this.relayService.deliverToRelays(src, updateAct); // Deliver Move activity to the followers of the old account const moveAct = this.apRendererService.addContext(this.apRendererService.renderMove(src, dst)); await this.apDeliverManagerService.deliverToFollowers(src, moveAct); // Publish meUpdated event const iObj = await this.userEntityService.pack(src.id, src, { detail: true, includeSecrets: true }); this.globalEventService.publishMainStream(src.id, 'meUpdated', iObj); // Move! await this.move(src, dst); return iObj; } /** * Create an alias of an old remote account. * * The user's new profile will be published to the followers. */ @bindThis public async createAlias(me: LocalUser, updates: Partial): Promise { await this.usersRepository.update(me.id, updates); me = Object.assign(me, updates); // Publish meUpdated event const iObj = await this.userEntityService.pack(me.id, me, { detail: true, includeSecrets: true, }); this.globalEventService.publishMainStream(me.id, 'meUpdated', iObj); if (me.isLocked === false) { await this.userFollowingService.acceptAllFollowRequests(me); } this.accountUpdateService.publishToFollowers(me.id); return iObj; } @bindThis public async move(src: User, dst: User): Promise { // Copy blockings and mutings, and update lists await Promise.all([ this.copyBlocking(src, dst), this.copyMutings(src, dst), this.updateLists(src, dst), ]); // follow the new account and unfollow the old one const proxy = await this.proxyAccountService.fetch(); const followings = await this.followingsRepository.find({ relations: { follower: true, }, where: { followeeId: src.id, followerHost: IsNull(), // follower is local followerId: proxy ? Not(proxy.id) : undefined, }, }); const followJobs: RelationshipJobData[] = []; for (const following of followings) { if (!following.follower) continue; followJobs.push({ from: { id: following.follower.id }, to: { id: dst.id } }); } // Decrease following count instead of unfollowing. await this.adjustFollowingCounts(followJobs.map(job => job.from.id), src); // Should be queued because this can cause a number of follow per one move. this.queueService.createFollowJob(followJobs); } @bindThis public async copyBlocking(src: ThinUser, dst: ThinUser): Promise { // Followers shouldn't overlap with blockers, but the destination account, different from the blockee (i.e., old account), may have followed the local user before moving. // So block the destination account here. const blockings = await this.blockingsRepository.find({ // FIXME: might be expensive relations: { blocker: true }, where: { blockeeId: src.id } }); // reblock the destination account const blockJobs: RelationshipJobData[] = []; for (const blocking of blockings) { if (!blocking.blocker) continue; blockJobs.push({ from: { id: blocking.blocker.id }, to: { id: dst.id } }); } // no need to unblock the old account because it may be still functional this.queueService.createBlockJob(blockJobs); } @bindThis public async copyMutings(src: ThinUser, dst: ThinUser): Promise { // Insert new mutings with the same values except mutee const mutings = await this.mutingsRepository.findBy({ muteeId: src.id, expiresAt: MoreThan(new Date()) }); const muteIds = mutings.map(mute => mute.id); if (muteIds.length > 0) { await this.mutingsRepository.update({ id: In(muteIds) }, { muteeId: dst.id }); for (const muterId of mutings.map(mute => mute.muterId)) { this.cacheService.userMutingsCache.refresh(muterId); } } } @bindThis public async updateLists(src: ThinUser, dst: User): Promise { // Return if there is no list to be updated. const exists = await this.userListJoiningsRepository.exist({ where: { userId: src.id, }, }); if (!exists) return; await this.userListJoiningsRepository.update( { userId: src.id }, { userId: dst.id } ); // Have the proxy account follow the new account in the same way as UserListService.push if (this.userEntityService.isRemoteUser(dst)) { const proxy = await this.proxyAccountService.fetch(); if (proxy) { this.queueService.createFollowJob([{ from: { id: proxy.id }, to: { id: dst.id } }]); } } } @bindThis public getUserUri(user: User): string { return this.userEntityService.isRemoteUser(user) ? user.uri : `${this.config.url}/users/${user.id}`; } @bindThis private async adjustFollowingCounts(localFollowerIds: string[], oldAccount: User): Promise { if (localFollowerIds.length === 0) return; // Set the old account's following and followers counts to 0. await this.usersRepository.update({ id: oldAccount.id }, { followersCount: 0, followingCount: 0 }); // Decrease following counts of local followers by 1. await this.usersRepository.decrement({ id: In(localFollowerIds) }, 'followingCount', 1); // Decrease follower counts of local followees by 1. const oldFollowings = await this.followingsRepository.findBy({ followerId: oldAccount.id }); if (oldFollowings.length > 0) { await this.usersRepository.decrement({ id: In(oldFollowings.map(following => following.followeeId)) }, 'followersCount', 1); } // Update instance stats by decreasing remote followers count by the number of local followers who were following the old account. if (this.userEntityService.isRemoteUser(oldAccount)) { this.federatedInstanceService.fetch(oldAccount.host).then(async i => { this.instancesRepository.decrement({ id: i.id }, 'followersCount', localFollowerIds.length); if ((await this.metaService.fetch()).enableChartsForFederatedInstances) { this.instanceChart.updateFollowers(i.host, false); } }); } // FIXME: expensive? for (const followerId of localFollowerIds) { this.perUserFollowingChart.update({ id: followerId, host: null }, oldAccount, false); } } }