X-Git-Url: https://git.immae.eu/?a=blobdiff_plain;f=server%2Flib%2Factivitypub%2Fprocess%2Fprocess-follow.ts;h=9009c646961b49de05af9a8cfc802dc1ff05a11a;hb=7d9ba5c08999c6482f0bc5e0c09c6f55b7724090;hp=320dc113814f74757c52f6e5fa753bf8f291ead1;hpb=40ff57078e15d5b86ee6b71e198b95d3feb78eaf;p=github%2FChocobozzz%2FPeerTube.git diff --git a/server/lib/activitypub/process/process-follow.ts b/server/lib/activitypub/process/process-follow.ts index 320dc1138..9009c6469 100644 --- a/server/lib/activitypub/process/process-follow.ts +++ b/server/lib/activitypub/process/process-follow.ts @@ -1,16 +1,25 @@ -import { ActivityFollow } from '../../../../shared/models/activitypub/activity' -import { retryTransactionWrapper } from '../../../helpers' -import { database as db } from '../../../initializers' -import { AccountInstance } from '../../../models/account/account-interface' +import { getServerActor } from '@server/models/application/application' +import { ActivityFollow } from '../../../../shared/models/activitypub' +import { getAPId } from '../../../helpers/activitypub' +import { retryTransactionWrapper } from '../../../helpers/database-utils' import { logger } from '../../../helpers/logger' -import { sendAccept } from '../send/send-accept' -import { getOrCreateAccountAndServer } from '../account' +import { CONFIG } from '../../../initializers/config' +import { sequelizeTypescript } from '../../../initializers/database' +import { ActorModel } from '../../../models/actor/actor' +import { ActorFollowModel } from '../../../models/actor/actor-follow' +import { APProcessorOptions } from '../../../types/activitypub-processor.model' +import { MActorFollowActors, MActorSignature } from '../../../types/models' +import { Notifier } from '../../notifier' +import { autoFollowBackIfNeeded } from '../follow' +import { sendAccept, sendReject } from '../send' -async function processFollowActivity (activity: ActivityFollow) { - const activityObject = activity.object - const account = await getOrCreateAccountAndServer(activity.actor) +async function processFollowActivity (options: APProcessorOptions) { + const { activity, byActor } = options - return processFollow(account, activityObject) + const activityId = activity.id + const objectId = getAPId(activity.object) + + return retryTransactionWrapper(processFollow, byActor, activityId, objectId) } // --------------------------------------------------------------------------- @@ -21,46 +30,80 @@ export { // --------------------------------------------------------------------------- -function processFollow (account: AccountInstance, targetAccountURL: string) { - const options = { - arguments: [ account, targetAccountURL ], - errorMessage: 'Cannot follow with many retries.' - } +async function processFollow (byActor: MActorSignature, activityId: string, targetActorURL: string) { + const { actorFollow, created, isFollowingInstance, targetActor } = await sequelizeTypescript.transaction(async t => { + const targetActor = await ActorModel.loadByUrlAndPopulateAccountAndChannel(targetActorURL, t) - return retryTransactionWrapper(follow, options) -} + if (!targetActor) throw new Error('Unknown actor') + if (targetActor.isOwned() === false) throw new Error('This is not a local actor.') -async function follow (account: AccountInstance, targetAccountURL: string) { - await db.sequelize.transaction(async t => { - const targetAccount = await db.Account.loadByUrl(targetAccountURL, t) + const serverActor = await getServerActor() + const isFollowingInstance = targetActor.id === serverActor.id - if (!targetAccount) throw new Error('Unknown account') - if (targetAccount.isOwned() === false) throw new Error('This is not a local account.') + if (isFollowingInstance && CONFIG.FOLLOWERS.INSTANCE.ENABLED === false) { + logger.info('Rejecting %s because instance followers are disabled.', targetActor.url) - const [ accountFollow ] = await db.AccountFollow.findOrCreate({ + await sendReject(activityId, byActor, targetActor) + + return { actorFollow: undefined as MActorFollowActors } + } + + const [ actorFollow, created ] = await ActorFollowModel.findOrCreate({ where: { - accountId: account.id, - targetAccountId: targetAccount.id + actorId: byActor.id, + targetActorId: targetActor.id }, defaults: { - accountId: account.id, - targetAccountId: targetAccount.id, - state: 'accepted' + actorId: byActor.id, + targetActorId: targetActor.id, + url: activityId, + + state: CONFIG.FOLLOWERS.INSTANCE.MANUAL_APPROVAL + ? 'pending' + : 'accepted' }, transaction: t }) - if (accountFollow.state !== 'accepted') { - accountFollow.state = 'accepted' - await accountFollow.save({ transaction: t }) + // Set the follow as accepted if the remote actor follows a channel or account + // Or if the instance automatically accepts followers + if (actorFollow.state !== 'accepted' && (isFollowingInstance === false || CONFIG.FOLLOWERS.INSTANCE.MANUAL_APPROVAL === false)) { + actorFollow.state = 'accepted' + + await actorFollow.save({ transaction: t }) } - accountFollow.AccountFollower = account - accountFollow.AccountFollowing = targetAccount + // Before PeerTube V3 we did not save the follow ID. Try to fix these old follows + if (!actorFollow.url) { + actorFollow.url = activityId + await actorFollow.save({ transaction: t }) + } - // Target sends to account he accepted the follow request - return sendAccept(accountFollow, t) + actorFollow.ActorFollower = byActor + actorFollow.ActorFollowing = targetActor + + // Target sends to actor he accepted the follow request + if (actorFollow.state === 'accepted') { + await sendAccept(actorFollow) + await autoFollowBackIfNeeded(actorFollow) + } + + return { actorFollow, created, isFollowingInstance, targetActor } }) - logger.info('Account uuid %s is followed by account %s.', account.url, targetAccountURL) + // Rejected + if (!actorFollow) return + + if (created) { + const follower = await ActorModel.loadFull(byActor.id) + const actorFollowFull = Object.assign(actorFollow, { ActorFollowing: targetActor, ActorFollower: follower }) + + if (isFollowingInstance) { + Notifier.Instance.notifyOfNewInstanceFollow(actorFollowFull) + } else { + Notifier.Instance.notifyOfNewUserFollow(actorFollowFull) + } + } + + logger.info('Actor %s is followed by actor %s.', targetActorURL, byActor.url) }