import { ActorModel } from '../../../models/activitypub/actor'
import { ActorFollowModel } from '../../../models/activitypub/actor-follow'
import { JobQueue } from '../../job-queue'
-import { VideoModel } from '../../../models/video/video'
import { getActorsInvolvedInVideo, getAudienceFromFollowersOf, getRemoteVideoAudience } from '../audience'
-import { getServerActor } from '../../../helpers/utils'
+import { afterCommitIfTransaction } from '../../../helpers/database-utils'
+import { MActor, MActorId, MActorLight, MActorWithInboxes, MVideoAccountLight, MVideoId, MVideoImmutable } from '../../../typings/models'
+import { getServerActor } from '@server/models/application/application'
+import { ContextType } from '@shared/models/activitypub/context'
async function sendVideoRelatedActivity (activityBuilder: (audience: ActivityAudience) => Activity, options: {
- byActor: ActorModel,
- video: VideoModel,
+ byActor: MActorLight
+ video: MVideoImmutable | MVideoAccountLight
transaction?: Transaction
+ contextType?: ContextType
}) {
- const actorsInvolvedInVideo = await getActorsInvolvedInVideo(options.video, options.transaction)
+ const { byActor, video, transaction, contextType } = options
+
+ const actorsInvolvedInVideo = await getActorsInvolvedInVideo(video, transaction)
// Send to origin
- if (options.video.isOwned() === false) {
- const audience = getRemoteVideoAudience(options.video, actorsInvolvedInVideo)
+ if (video.isOwned() === false) {
+ const accountActor = (video as MVideoAccountLight).VideoChannel?.Account?.Actor || await ActorModel.loadAccountActorByVideoId(video.id)
+
+ const audience = getRemoteVideoAudience(accountActor, actorsInvolvedInVideo)
const activity = activityBuilder(audience)
- return unicastTo(activity, options.byActor, options.video.VideoChannel.Account.Actor.sharedInboxUrl)
+ return afterCommitIfTransaction(transaction, () => {
+ return unicastTo(activity, byActor, accountActor.getSharedInbox(), contextType)
+ })
}
// Send to followers
const audience = getAudienceFromFollowersOf(actorsInvolvedInVideo)
const activity = activityBuilder(audience)
- const actorsException = [ options.byActor ]
- return broadcastToFollowers(activity, options.byActor, actorsInvolvedInVideo, options.transaction, actorsException)
+ const actorsException = [ byActor ]
+
+ return broadcastToFollowers(activity, byActor, actorsInvolvedInVideo, transaction, actorsException, contextType)
}
async function forwardVideoRelatedActivity (
activity: Activity,
t: Transaction,
- followersException: ActorModel[] = [],
- video: VideoModel
+ followersException: MActorWithInboxes[],
+ video: MVideoId
) {
// Mastodon does not add our announces in audience, so we forward to them manually
const additionalActors = await getActorsInvolvedInVideo(video, t)
async function forwardActivity (
activity: Activity,
t: Transaction,
- followersException: ActorModel[] = [],
+ followersException: MActorWithInboxes[] = [],
additionalFollowerUrls: string[] = []
) {
logger.info('Forwarding activity %s.', activity.id)
uris,
body: activity
}
- return JobQueue.Instance.createJob({ type: 'activitypub-http-broadcast', payload })
+ return afterCommitIfTransaction(t, () => JobQueue.Instance.createJob({ type: 'activitypub-http-broadcast', payload }))
}
async function broadcastToFollowers (
data: any,
- byActor: ActorModel,
- toFollowersOf: ActorModel[],
+ byActor: MActorId,
+ toFollowersOf: MActorId[],
t: Transaction,
- actorsException: ActorModel[] = []
+ actorsException: MActorWithInboxes[] = [],
+ contextType?: ContextType
) {
const uris = await computeFollowerUris(toFollowersOf, actorsException, t)
- return broadcastTo(uris, data, byActor)
+
+ return afterCommitIfTransaction(t, () => broadcastTo(uris, data, byActor, contextType))
}
async function broadcastToActors (
data: any,
- byActor: ActorModel,
- toActors: ActorModel[],
- actorsException: ActorModel[] = []
+ byActor: MActorId,
+ toActors: MActor[],
+ t?: Transaction,
+ actorsException: MActorWithInboxes[] = [],
+ contextType?: ContextType
) {
const uris = await computeUris(toActors, actorsException)
- return broadcastTo(uris, data, byActor)
+ return afterCommitIfTransaction(t, () => broadcastTo(uris, data, byActor, contextType))
}
-async function broadcastTo (uris: string[], data: any, byActor: ActorModel) {
+function broadcastTo (uris: string[], data: any, byActor: MActorId, contextType?: ContextType) {
if (uris.length === 0) return undefined
logger.debug('Creating broadcast job.', { uris })
const payload = {
uris,
signatureActorId: byActor.id,
- body: data
+ body: data,
+ contextType
}
return JobQueue.Instance.createJob({ type: 'activitypub-http-broadcast', payload })
}
-async function unicastTo (data: any, byActor: ActorModel, toActorUrl: string) {
+function unicastTo (data: any, byActor: MActorId, toActorUrl: string, contextType?: ContextType) {
logger.debug('Creating unicast job.', { uri: toActorUrl })
const payload = {
uri: toActorUrl,
signatureActorId: byActor.id,
- body: data
+ body: data,
+ contextType
}
- return JobQueue.Instance.createJob({ type: 'activitypub-http-unicast', payload })
+ JobQueue.Instance.createJob({ type: 'activitypub-http-unicast', payload })
}
// ---------------------------------------------------------------------------
// ---------------------------------------------------------------------------
-async function computeFollowerUris (toFollowersOf: ActorModel[], actorsException: ActorModel[], t: Transaction) {
+async function computeFollowerUris (toFollowersOf: MActorId[], actorsException: MActorWithInboxes[], t: Transaction) {
const toActorFollowerIds = toFollowersOf.map(a => a.id)
const result = await ActorFollowModel.listAcceptedFollowerSharedInboxUrls(toActorFollowerIds, t)
const sharedInboxesException = await buildSharedInboxesException(actorsException)
- return result.data.filter(sharedInbox => sharedInboxesException.indexOf(sharedInbox) === -1)
+ return result.data.filter(sharedInbox => sharedInboxesException.includes(sharedInbox) === false)
}
-async function computeUris (toActors: ActorModel[], actorsException: ActorModel[] = []) {
+async function computeUris (toActors: MActor[], actorsException: MActorWithInboxes[] = []) {
const serverActor = await getServerActor()
const targetUrls = toActors
.filter(a => a.id !== serverActor.id) // Don't send to ourselves
- .map(a => a.sharedInboxUrl || a.inboxUrl)
+ .map(a => a.getSharedInbox())
const toActorSharedInboxesSet = new Set(targetUrls)
const sharedInboxesException = await buildSharedInboxesException(actorsException)
return Array.from(toActorSharedInboxesSet)
- .filter(sharedInbox => sharedInboxesException.indexOf(sharedInbox) === -1)
+ .filter(sharedInbox => sharedInboxesException.includes(sharedInbox) === false)
}
-async function buildSharedInboxesException (actorsException: ActorModel[]) {
+async function buildSharedInboxesException (actorsException: MActorWithInboxes[]) {
const serverActor = await getServerActor()
return actorsException
- .map(f => f.sharedInboxUrl || f.inboxUrl)
+ .map(f => f.getSharedInbox())
.concat([ serverActor.sharedInboxUrl ])
}