import { Transaction } from 'sequelize'
import { ActivityAudience, ActivityUpdate } from '../../../../shared/models/activitypub'
import { VideoPrivacy } from '../../../../shared/models/videos'
-import { ActorModel } from '../../../models/activitypub/actor'
+import { AccountModel } from '../../../models/account/account'
import { VideoModel } from '../../../models/video/video'
import { VideoShareModel } from '../../../models/video/video-share'
import { getUpdateActivityPubUrl } from '../url'
-import { audiencify, broadcastToFollowers, getAudience } from './misc'
+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 '../../../typings/models'
+import { getServerActor } from '@server/models/application/application'
-async function sendUpdateVideo (video: VideoModel, t: Transaction) {
- const byActor = video.VideoChannel.Account.Actor
+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 = await getAudience(byActor, t, video.privacy === VideoPrivacy.PUBLIC)
+ 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)
- const data = await updateActivityData(url, byActor, videoObject, t, audience)
+ 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)
+ }
- const actorsInvolved = await VideoShareModel.loadActorsByShare(video.id, t)
actorsInvolved.push(byActor)
- return broadcastToFollowers(data, byActor, actorsInvolved, t)
+ 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 video = await VideoModel.loadAndPopulateAccountAndServerAndTags(redundancyModel.getVideo().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 {
- sendUpdateVideo
+ sendUpdateActor,
+ sendUpdateVideo,
+ sendUpdateCacheFile,
+ sendUpdateVideoPlaylist
}
// ---------------------------------------------------------------------------
-async function updateActivityData (
- url: string,
- byActor: ActorModel,
- object: any,
- t: Transaction,
- audience?: ActivityAudience
-): Promise<ActivityUpdate> {
- if (!audience) {
- audience = await getAudience(byActor, t)
- }
+function buildUpdateActivity (url: string, byActor: MActorLight, object: any, audience?: ActivityAudience): ActivityUpdate {
+ if (!audience) audience = getAudience(byActor)
- return audiencify({
- type: 'Update',
- id: url,
- actor: byActor.url,
- object: audiencify(object, audience)
- }, audience)
+ return audiencify(
+ {
+ type: 'Update' as 'Update',
+ id: url,
+ actor: byActor.url,
+ object: audiencify(object, audience)
+ },
+ audience
+ )
}