142 lines
5.3 KiB
TypeScript
142 lines
5.3 KiB
TypeScript
import { Transaction } from 'sequelize'
|
|
import { ActivityAudience, ActivityUpdate } from '../../../../shared/models/activitypub'
|
|
import { VideoPrivacy } from '../../../../shared/models/videos'
|
|
import { AccountModel } from '../../../models/account/account'
|
|
import { VideoModel } from '../../../models/video/video'
|
|
import { VideoShareModel } from '../../../models/video/video-share'
|
|
import { getUpdateActivityPubUrl } from '../url'
|
|
import { broadcastToFollowers, sendVideoRelatedActivity } from './utils'
|
|
import { audiencify, getActorsInvolvedInVideo, getAudience } from '../audience'
|
|
import { logger } from '../../../helpers/logger'
|
|
import { VideoPlaylistPrivacy } from '../../../../shared/models/videos/playlist/video-playlist-privacy.model'
|
|
import {
|
|
MAccountDefault,
|
|
MActor,
|
|
MActorLight,
|
|
MChannelDefault,
|
|
MVideoAP,
|
|
MVideoAPWithoutCaption,
|
|
MVideoPlaylistFull,
|
|
MVideoRedundancyVideo
|
|
} from '../../../types/models'
|
|
import { getServerActor } from '@server/models/application/application'
|
|
|
|
async function sendUpdateVideo (videoArg: MVideoAPWithoutCaption, t: Transaction, overrodeByActor?: MActor) {
|
|
const video = videoArg as MVideoAP
|
|
|
|
if (!video.hasPrivacyForFederation()) return undefined
|
|
|
|
logger.info('Creating job to update video %s.', video.url)
|
|
|
|
const byActor = overrodeByActor || video.VideoChannel.Account.Actor
|
|
|
|
const url = getUpdateActivityPubUrl(video.url, video.updatedAt.toISOString())
|
|
|
|
// Needed to build the AP object
|
|
if (!video.VideoCaptions) {
|
|
video.VideoCaptions = await video.$get('VideoCaptions', { transaction: t })
|
|
}
|
|
|
|
const videoObject = video.toActivityPubObject()
|
|
const audience = getAudience(byActor, video.privacy === VideoPrivacy.PUBLIC)
|
|
|
|
const updateActivity = buildUpdateActivity(url, byActor, videoObject, audience)
|
|
|
|
const actorsInvolved = await getActorsInvolvedInVideo(video, t)
|
|
if (overrodeByActor) actorsInvolved.push(overrodeByActor)
|
|
|
|
return broadcastToFollowers(updateActivity, byActor, actorsInvolved, t)
|
|
}
|
|
|
|
async function sendUpdateActor (accountOrChannel: MChannelDefault | MAccountDefault, t: Transaction) {
|
|
const byActor = accountOrChannel.Actor
|
|
|
|
logger.info('Creating job to update actor %s.', byActor.url)
|
|
|
|
const url = getUpdateActivityPubUrl(byActor.url, byActor.updatedAt.toISOString())
|
|
const accountOrChannelObject = (accountOrChannel as any).toActivityPubObject() // FIXME: typescript bug?
|
|
const audience = getAudience(byActor)
|
|
const updateActivity = buildUpdateActivity(url, byActor, accountOrChannelObject, audience)
|
|
|
|
let actorsInvolved: MActor[]
|
|
if (accountOrChannel instanceof AccountModel) {
|
|
// Actors that shared my videos are involved too
|
|
actorsInvolved = await VideoShareModel.loadActorsWhoSharedVideosOf(byActor.id, t)
|
|
} else {
|
|
// Actors that shared videos of my channel are involved too
|
|
actorsInvolved = await VideoShareModel.loadActorsByVideoChannel(accountOrChannel.id, t)
|
|
}
|
|
|
|
actorsInvolved.push(byActor)
|
|
|
|
return broadcastToFollowers(updateActivity, byActor, actorsInvolved, t)
|
|
}
|
|
|
|
async function sendUpdateCacheFile (byActor: MActorLight, redundancyModel: MVideoRedundancyVideo) {
|
|
logger.info('Creating job to update cache file %s.', redundancyModel.url)
|
|
|
|
const associatedVideo = redundancyModel.getVideo()
|
|
if (!associatedVideo) {
|
|
logger.warn('Cannot send update activity for redundancy %s: no video files associated.', redundancyModel.url)
|
|
return
|
|
}
|
|
|
|
const video = await VideoModel.loadAndPopulateAccountAndServerAndTags(associatedVideo.id)
|
|
|
|
const activityBuilder = (audience: ActivityAudience) => {
|
|
const redundancyObject = redundancyModel.toActivityPubObject()
|
|
const url = getUpdateActivityPubUrl(redundancyModel.url, redundancyModel.updatedAt.toISOString())
|
|
|
|
return buildUpdateActivity(url, byActor, redundancyObject, audience)
|
|
}
|
|
|
|
return sendVideoRelatedActivity(activityBuilder, { byActor, video, contextType: 'CacheFile' })
|
|
}
|
|
|
|
async function sendUpdateVideoPlaylist (videoPlaylist: MVideoPlaylistFull, t: Transaction) {
|
|
if (videoPlaylist.privacy === VideoPlaylistPrivacy.PRIVATE) return undefined
|
|
|
|
const byActor = videoPlaylist.OwnerAccount.Actor
|
|
|
|
logger.info('Creating job to update video playlist %s.', videoPlaylist.url)
|
|
|
|
const url = getUpdateActivityPubUrl(videoPlaylist.url, videoPlaylist.updatedAt.toISOString())
|
|
|
|
const object = await videoPlaylist.toActivityPubObject(null, t)
|
|
const audience = getAudience(byActor, videoPlaylist.privacy === VideoPlaylistPrivacy.PUBLIC)
|
|
|
|
const updateActivity = buildUpdateActivity(url, byActor, object, audience)
|
|
|
|
const serverActor = await getServerActor()
|
|
const toFollowersOf = [ byActor, serverActor ]
|
|
|
|
if (videoPlaylist.VideoChannel) toFollowersOf.push(videoPlaylist.VideoChannel.Actor)
|
|
|
|
return broadcastToFollowers(updateActivity, byActor, toFollowersOf, t)
|
|
}
|
|
|
|
// ---------------------------------------------------------------------------
|
|
|
|
export {
|
|
sendUpdateActor,
|
|
sendUpdateVideo,
|
|
sendUpdateCacheFile,
|
|
sendUpdateVideoPlaylist
|
|
}
|
|
|
|
// ---------------------------------------------------------------------------
|
|
|
|
function buildUpdateActivity (url: string, byActor: MActorLight, object: any, audience?: ActivityAudience): ActivityUpdate {
|
|
if (!audience) audience = getAudience(byActor)
|
|
|
|
return audiencify(
|
|
{
|
|
type: 'Update' as 'Update',
|
|
id: url,
|
|
actor: byActor.url,
|
|
object: audiencify(object, audience)
|
|
},
|
|
audience
|
|
)
|
|
}
|