PeerTube/server/lib/activitypub/videos/shared/creator.ts

91 lines
3.3 KiB
TypeScript
Raw Normal View History

2021-06-02 10:41:46 +02:00
2021-06-03 16:56:42 +02:00
import { logger, loggerTagsFactory, LoggerTagsFn } from '@server/helpers/logger'
2021-06-02 10:41:46 +02:00
import { sequelizeTypescript } from '@server/initializers/database'
2023-03-16 10:36:33 +01:00
import { Hooks } from '@server/lib/plugins/hooks'
2021-06-02 10:41:46 +02:00
import { autoBlacklistVideoIfNeeded } from '@server/lib/video-blacklist'
import { VideoModel } from '@server/models/video/video'
2021-06-02 11:54:29 +02:00
import { MThumbnail, MVideoFullLight, MVideoThumbnail } from '@server/types/models'
2021-06-02 10:41:46 +02:00
import { VideoObject } from '@shared/models'
import { APVideoAbstractBuilder } from './abstract-builder'
import { getVideoAttributesFromObject } from './object-to-model-attributes'
export class APVideoCreator extends APVideoAbstractBuilder {
2021-06-03 16:56:42 +02:00
protected lTags: LoggerTagsFn
2021-06-02 10:41:46 +02:00
2021-06-02 11:54:29 +02:00
constructor (protected readonly videoObject: VideoObject) {
2021-06-02 10:41:46 +02:00
super()
2021-06-03 16:56:42 +02:00
this.lTags = loggerTagsFactory('ap', 'video', 'create', this.videoObject.uuid, this.videoObject.id)
2021-06-02 10:41:46 +02:00
}
async create (waitThumbnail = false) {
2021-06-03 16:56:42 +02:00
logger.debug('Adding remote video %s.', this.videoObject.id, this.lTags())
2021-06-02 10:41:46 +02:00
2021-06-02 11:54:29 +02:00
const channelActor = await this.getOrCreateVideoChannelFromVideoObject()
const channel = channelActor.VideoChannel
2021-06-15 08:36:39 +02:00
const videoData = getVideoAttributesFromObject(channel, this.videoObject, this.videoObject.to)
const video = VideoModel.build({ ...videoData, likes: 0, dislikes: 0 }) as MVideoThumbnail
2021-06-02 10:41:46 +02:00
const promiseThumbnail = this.tryToGenerateThumbnail(video)
let thumbnailModel: MThumbnail
if (waitThumbnail === true) {
thumbnailModel = await promiseThumbnail
}
const { autoBlacklisted, videoCreated } = await sequelizeTypescript.transaction(async t => {
try {
const videoCreated = await video.save({ transaction: t }) as MVideoFullLight
2021-06-02 11:54:29 +02:00
videoCreated.VideoChannel = channel
2021-06-02 10:41:46 +02:00
if (thumbnailModel) await videoCreated.addAndSaveThumbnail(thumbnailModel, t)
await this.setPreview(videoCreated, t)
await this.setWebTorrentFiles(videoCreated, t)
await this.setStreamingPlaylists(videoCreated, t)
await this.setTags(videoCreated, t)
await this.setTrackers(videoCreated, t)
await this.insertOrReplaceCaptions(videoCreated, t)
await this.insertOrReplaceLive(videoCreated, t)
// We added a video in this channel, set it as updated
2021-06-02 11:54:29 +02:00
await channel.setAsUpdated(t)
2021-06-02 10:41:46 +02:00
const autoBlacklisted = await autoBlacklistVideoIfNeeded({
video: videoCreated,
user: undefined,
isRemote: true,
isNew: true,
transaction: t
})
2021-06-03 16:56:42 +02:00
logger.info('Remote video with uuid %s inserted.', this.videoObject.uuid, this.lTags())
2021-06-02 10:41:46 +02:00
2023-03-16 10:36:33 +01:00
Hooks.runAction('action:activity-pub.remote-video.created', { video: videoCreated, videoAPObject: this.videoObject })
2021-06-02 10:41:46 +02:00
return { autoBlacklisted, videoCreated }
} catch (err) {
// FIXME: Use rollback hook when https://github.com/sequelize/sequelize/pull/13038 is released
if (thumbnailModel) await thumbnailModel.removeThumbnail()
throw err
}
})
if (waitThumbnail === false) {
// Error is already caught above
// eslint-disable-next-line @typescript-eslint/no-floating-promises
promiseThumbnail.then(thumbnailModel => {
if (!thumbnailModel) return
thumbnailModel = videoCreated.id
return thumbnailModel.save()
})
}
return { autoBlacklisted, videoCreated }
}
}