import { logger } from '../../helpers/logger' import { AbstractScheduler } from './abstract-scheduler' import { INSTANCES_INDEX, SCHEDULER_INTERVALS_MS, SERVER_ACTOR_NAME } from '../../initializers/constants' import { CONFIG } from '../../initializers/config' import { chunk } from 'lodash' import { doRequest } from '@server/helpers/requests' import { ActorFollowModel } from '@server/models/activitypub/actor-follow' import { JobQueue } from '@server/lib/job-queue' import { getServerActor } from '@server/helpers/utils' export class AutoFollowIndexInstances extends AbstractScheduler { private static instance: AbstractScheduler protected schedulerIntervalMs = SCHEDULER_INTERVALS_MS.autoFollowIndexInstances private lastCheck: Date private constructor () { super() } protected async internalExecute () { return this.autoFollow() } private async autoFollow () { if (CONFIG.FOLLOWINGS.INSTANCE.AUTO_FOLLOW_INDEX.ENABLED === false) return const indexUrl = CONFIG.FOLLOWINGS.INSTANCE.AUTO_FOLLOW_INDEX.INDEX_URL logger.info('Auto follow instances of index %s.', indexUrl) try { const serverActor = await getServerActor() const uri = indexUrl + INSTANCES_INDEX.HOSTS_PATH const qs = this.lastCheck ? { since: this.lastCheck.toISOString() } : {} this.lastCheck = new Date() const { body } = await doRequest({ uri, qs, json: true }) const hosts: string[] = body.data.map(o => o.host) const chunks = chunk(hosts, 20) for (const chunk of chunks) { const unfollowedHosts = await ActorFollowModel.keepUnfollowedInstance(chunk) for (const unfollowedHost of unfollowedHosts) { const payload = { host: unfollowedHost, name: SERVER_ACTOR_NAME, followerActorId: serverActor.id, isAutoFollow: true } await JobQueue.Instance.createJob({ type: 'activitypub-follow', payload }) .catch(err => logger.error('Cannot create follow job for %s.', unfollowedHost, err)) } } } catch (err) { logger.error('Cannot auto follow hosts of index %s.', indexUrl, { err }) } } static get Instance () { return this.instance || (this.instance = new this()) } }