import { Transaction } from 'sequelize'
-import { ActivityAudience, ActivityCreate } from '../../../../shared/models/activitypub'
-import { VideoPrivacy } from '../../../../shared/models/videos'
-import { ActorModel } from '../../../models/activitypub/actor'
-import { VideoModel } from '../../../models/video/video'
-import { VideoAbuseModel } from '../../../models/video/video-abuse'
+import { getServerActor } from '@server/models/application/application'
+import { ActivityAudience, ActivityCreate, ContextType, VideoPlaylistPrivacy, VideoPrivacy } from '@shared/models'
+import { logger, loggerTagsFactory } from '../../../helpers/logger'
import { VideoCommentModel } from '../../../models/video/video-comment'
-import { getVideoAbuseActivityPubUrl, getVideoDislikeActivityPubUrl, getVideoViewActivityPubUrl } from '../url'
-import { broadcastToActors, broadcastToFollowers, sendVideoRelatedActivity, unicastTo } from './utils'
-import { audiencify, getActorsInvolvedInVideo, getAudience, getAudienceFromFollowersOf, getVideoCommentAudience } from '../audience'
-import { logger } from '../../../helpers/logger'
-import { VideoRedundancyModel } from '../../../models/redundancy/video-redundancy'
-
-async function sendCreateVideo (video: VideoModel, t: Transaction) {
- if (video.privacy === VideoPrivacy.PRIVATE) return undefined
-
- logger.info('Creating job to send video creation of %s.', video.url)
+import {
+ MActorLight,
+ MCommentOwnerVideo,
+ MLocalVideoViewerWithWatchSections,
+ MVideoAccountLight,
+ MVideoAP,
+ MVideoPlaylistFull,
+ MVideoRedundancyFileVideo,
+ MVideoRedundancyStreamingPlaylistVideo
+} from '../../../types/models'
+import { audiencify, getAudience } from '../audience'
+import {
+ broadcastToActors,
+ broadcastToFollowers,
+ getActorsInvolvedInVideo,
+ getAudienceFromFollowersOf,
+ getVideoCommentAudience,
+ sendVideoActivityToOrigin,
+ sendVideoRelatedActivity,
+ unicastTo
+} from './shared'
+
+const lTags = loggerTagsFactory('ap', 'create')
+
+async function sendCreateVideo (video: MVideoAP, transaction: Transaction) {
+ if (!video.hasPrivacyForFederation()) return undefined
+
+ logger.info('Creating job to send video creation of %s.', video.url, lTags(video.uuid))
const byActor = video.VideoChannel.Account.Actor
const videoObject = video.toActivityPubObject()
const audience = getAudience(byActor, video.privacy === VideoPrivacy.PUBLIC)
const createActivity = buildCreateActivity(video.url, byActor, videoObject, audience)
- return broadcastToFollowers(createActivity, byActor, [ byActor ], t)
+ return broadcastToFollowers({
+ data: createActivity,
+ byActor,
+ toFollowersOf: [ byActor ],
+ transaction,
+ contextType: 'Video'
+ })
}
-async function sendVideoAbuse (byActor: ActorModel, videoAbuse: VideoAbuseModel, video: VideoModel) {
- if (!video.VideoChannel.Account.Actor.serverId) return // Local
+async function sendCreateCacheFile (
+ byActor: MActorLight,
+ video: MVideoAccountLight,
+ fileRedundancy: MVideoRedundancyStreamingPlaylistVideo | MVideoRedundancyFileVideo
+) {
+ logger.info('Creating job to send file cache of %s.', fileRedundancy.url, lTags(video.uuid))
+
+ return sendVideoRelatedCreateActivity({
+ byActor,
+ video,
+ url: fileRedundancy.url,
+ object: fileRedundancy.toActivityPubObject(),
+ contextType: 'CacheFile'
+ })
+}
- const url = getVideoAbuseActivityPubUrl(videoAbuse)
+async function sendCreateWatchAction (stats: MLocalVideoViewerWithWatchSections, transaction: Transaction) {
+ logger.info('Creating job to send create watch action %s.', stats.url, lTags(stats.uuid))
- logger.info('Creating job to send video abuse %s.', url)
+ const byActor = await getServerActor()
- // Custom audience, we only send the abuse to the origin instance
- const audience = { to: [ video.VideoChannel.Account.Actor.url ], cc: [] }
- const createActivity = buildCreateActivity(url, byActor, videoAbuse.toActivityPubObject(), audience)
+ const activityBuilder = (audience: ActivityAudience) => {
+ return buildCreateActivity(stats.url, byActor, stats.toActivityPubObject(), audience)
+ }
- return unicastTo(createActivity, byActor, video.VideoChannel.Account.Actor.sharedInboxUrl)
+ return sendVideoActivityToOrigin(activityBuilder, { byActor, video: stats.Video, transaction, contextType: 'WatchAction' })
}
-async function sendCreateCacheFile (byActor: ActorModel, fileRedundancy: VideoRedundancyModel) {
- logger.info('Creating job to send file cache of %s.', fileRedundancy.url)
+async function sendCreateVideoPlaylist (playlist: MVideoPlaylistFull, transaction: Transaction) {
+ if (playlist.privacy === VideoPlaylistPrivacy.PRIVATE) return undefined
- const video = await VideoModel.loadAndPopulateAccountAndServerAndTags(fileRedundancy.VideoFile.Video.id)
- const redundancyObject = fileRedundancy.toActivityPubObject()
+ logger.info('Creating job to send create video playlist of %s.', playlist.url, lTags(playlist.uuid))
- return sendVideoRelatedCreateActivity({
+ const byActor = playlist.OwnerAccount.Actor
+ const audience = getAudience(byActor, playlist.privacy === VideoPlaylistPrivacy.PUBLIC)
+
+ const object = await playlist.toActivityPubObject(null, transaction)
+ const createActivity = buildCreateActivity(playlist.url, byActor, object, audience)
+
+ const serverActor = await getServerActor()
+ const toFollowersOf = [ byActor, serverActor ]
+
+ if (playlist.VideoChannel) toFollowersOf.push(playlist.VideoChannel.Actor)
+
+ return broadcastToFollowers({
+ data: createActivity,
byActor,
- video,
- url: fileRedundancy.url,
- object: redundancyObject
+ toFollowersOf,
+ transaction,
+ contextType: 'Playlist'
})
}
-async function sendCreateVideoComment (comment: VideoCommentModel, t: Transaction) {
+async function sendCreateVideoComment (comment: MCommentOwnerVideo, transaction: Transaction) {
logger.info('Creating job to send comment %s.', comment.url)
const isOrigin = comment.Video.isOwned()
const byActor = comment.Account.Actor
- const threadParentComments = await VideoCommentModel.listThreadParentComments(comment, t)
+ const threadParentComments = await VideoCommentModel.listThreadParentComments(comment, transaction)
const commentObject = comment.toActivityPubObject(threadParentComments)
- const actorsInvolvedInComment = await getActorsInvolvedInVideo(comment.Video, t)
+ const actorsInvolvedInComment = await getActorsInvolvedInVideo(comment.Video, transaction)
// Add the actor that commented too
actorsInvolvedInComment.push(byActor)
- const parentsCommentActors = threadParentComments.map(c => c.Account.Actor)
+ const parentsCommentActors = threadParentComments.filter(c => !c.isDeleted())
+ .map(c => c.Account.Actor)
let audience: ActivityAudience
if (isOrigin) {
// This was a reply, send it to the parent actors
const actorsException = [ byActor ]
- await broadcastToActors(createActivity, byActor, parentsCommentActors, actorsException)
+ await broadcastToActors({
+ data: createActivity,
+ byActor,
+ toActors: parentsCommentActors,
+ transaction,
+ actorsException,
+ contextType: 'Comment'
+ })
// Broadcast to our followers
- await broadcastToFollowers(createActivity, byActor, [ byActor ], t)
-
- // Send to actors involved in the comment
- if (isOrigin) return broadcastToFollowers(createActivity, byActor, actorsInvolvedInComment, t, actorsException)
-
- // Send to origin
- return unicastTo(createActivity, byActor, comment.Video.VideoChannel.Account.Actor.sharedInboxUrl)
-}
-
-async function sendCreateView (byActor: ActorModel, video: VideoModel, t: Transaction) {
- logger.info('Creating job to send view of %s.', video.url)
-
- const url = getVideoViewActivityPubUrl(byActor, video)
- const viewActivity = buildViewActivity(url, byActor, video)
-
- return sendVideoRelatedCreateActivity({
- // Use the server actor to send the view
+ await broadcastToFollowers({
+ data: createActivity,
byActor,
- video,
- url,
- object: viewActivity,
- transaction: t
+ toFollowersOf: [ byActor ],
+ transaction,
+ contextType: 'Comment'
})
-}
-async function sendCreateDislike (byActor: ActorModel, video: VideoModel, t: Transaction) {
- logger.info('Creating job to dislike %s.', video.url)
-
- const url = getVideoDislikeActivityPubUrl(byActor, video)
- const dislikeActivity = buildDislikeActivity(url, byActor, video)
+ // Send to actors involved in the comment
+ if (isOrigin) {
+ return broadcastToFollowers({
+ data: createActivity,
+ byActor,
+ toFollowersOf: actorsInvolvedInComment,
+ transaction,
+ actorsException,
+ contextType: 'Comment'
+ })
+ }
- return sendVideoRelatedCreateActivity({
- byActor,
- video,
- url,
- object: dislikeActivity,
- transaction: t
+ // Send to origin
+ return transaction.afterCommit(() => {
+ return unicastTo({
+ data: createActivity,
+ byActor,
+ toActorUrl: comment.Video.VideoChannel.Account.Actor.getSharedInbox(),
+ contextType: 'Comment'
+ })
})
}
-function buildCreateActivity (url: string, byActor: ActorModel, object: any, audience?: ActivityAudience): ActivityCreate {
+function buildCreateActivity (url: string, byActor: MActorLight, object: any, audience?: ActivityAudience): ActivityCreate {
if (!audience) audience = getAudience(byActor)
return audiencify(
)
}
-function buildDislikeActivity (url: string, byActor: ActorModel, video: VideoModel) {
- return {
- id: url,
- type: 'Dislike',
- actor: byActor.url,
- object: video.url
- }
-}
-
-function buildViewActivity (url: string, byActor: ActorModel, video: VideoModel) {
- return {
- id: url,
- type: 'View',
- actor: byActor.url,
- object: video.url
- }
-}
-
// ---------------------------------------------------------------------------
export {
sendCreateVideo,
- sendVideoAbuse,
buildCreateActivity,
- sendCreateView,
- sendCreateDislike,
- buildDislikeActivity,
sendCreateVideoComment,
- sendCreateCacheFile
+ sendCreateVideoPlaylist,
+ sendCreateCacheFile,
+ sendCreateWatchAction
}
// ---------------------------------------------------------------------------
async function sendVideoRelatedCreateActivity (options: {
- byActor: ActorModel,
- video: VideoModel,
- url: string,
- object: any,
+ byActor: MActorLight
+ video: MVideoAccountLight
+ url: string
+ object: any
+ contextType: ContextType
transaction?: Transaction
}) {
const activityBuilder = (audience: ActivityAudience) => {