+import { VideoModel } from '@server/models/video/video'
+import { MScheduleVideoUpdate } from '@server/types/models'
+import { VideoPrivacy, VideoState } from '@shared/models'
import { logger } from '../../helpers/logger'
-import { AbstractScheduler } from './abstract-scheduler'
-import { ScheduleVideoUpdateModel } from '../../models/video/schedule-video-update'
-import { retryTransactionWrapper } from '../../helpers/database-utils'
-import { federateVideoIfNeeded } from '../activitypub/videos'
import { SCHEDULER_INTERVALS_MS } from '../../initializers/constants'
-import { Notifier } from '../notifier'
import { sequelizeTypescript } from '../../initializers/database'
-import { MVideoFullLight } from '@server/types/models'
-import { VideoModel } from '@server/models/video/video'
+import { ScheduleVideoUpdateModel } from '../../models/video/schedule-video-update'
+import { Notifier } from '../notifier'
+import { addVideoJobsAfterUpdate } from '../video'
+import { VideoPathManager } from '../video-path-manager'
+import { setVideoPrivacy } from '../video-privacy'
+import { AbstractScheduler } from './abstract-scheduler'
export class UpdateVideosScheduler extends AbstractScheduler {
private static instance: AbstractScheduler
- protected schedulerIntervalMs = SCHEDULER_INTERVALS_MS.updateVideos
+ protected schedulerIntervalMs = SCHEDULER_INTERVALS_MS.UPDATE_VIDEOS
private constructor () {
super()
}
protected async internalExecute () {
- return retryTransactionWrapper(this.updateVideos.bind(this))
+ return this.updateVideos()
}
private async updateVideos () {
if (!await ScheduleVideoUpdateModel.areVideosToUpdate()) return undefined
const schedules = await ScheduleVideoUpdateModel.listVideosToUpdate()
- const publishedVideos: MVideoFullLight[] = []
for (const schedule of schedules) {
- await sequelizeTypescript.transaction(async t => {
- const video = await VideoModel.loadAndPopulateAccountAndServerAndTags(schedule.videoId, t)
+ const videoOnly = await VideoModel.load(schedule.videoId)
+ const mutexReleaser = await VideoPathManager.Instance.lockFiles(videoOnly.uuid)
- logger.info('Executing scheduled video update on %s.', video.uuid)
+ try {
+ const { video, published } = await this.updateAVideo(schedule)
- if (schedule.privacy) {
- const wasConfidentialVideo = video.isConfidential()
- const isNewVideo = video.isNewVideo(schedule.privacy)
+ if (published) Notifier.Instance.notifyOnVideoPublishedAfterScheduledUpdate(video)
+ } catch (err) {
+ logger.error('Cannot update video', { err })
+ }
- video.setPrivacy(schedule.privacy)
- await video.save({ transaction: t })
- await federateVideoIfNeeded(video, isNewVideo, t)
+ mutexReleaser()
+ }
+ }
- if (wasConfidentialVideo) {
- publishedVideos.push(video)
- }
+ private async updateAVideo (schedule: MScheduleVideoUpdate) {
+ let oldPrivacy: VideoPrivacy
+ let isNewVideo: boolean
+ let published = false
+
+ const video = await sequelizeTypescript.transaction(async t => {
+ const video = await VideoModel.loadFull(schedule.videoId, t)
+ if (video.state === VideoState.TO_TRANSCODE) return null
+
+ logger.info('Executing scheduled video update on %s.', video.uuid)
+
+ if (schedule.privacy) {
+ isNewVideo = video.isNewVideo(schedule.privacy)
+ oldPrivacy = video.privacy
+
+ setVideoPrivacy(video, schedule.privacy)
+ await video.save({ transaction: t })
+
+ if (oldPrivacy === VideoPrivacy.PRIVATE) {
+ published = true
}
+ }
- await schedule.destroy({ transaction: t })
- })
- }
+ await schedule.destroy({ transaction: t })
+
+ return video
+ })
- for (const v of publishedVideos) {
- Notifier.Instance.notifyOnNewVideoIfNeeded(v)
- Notifier.Instance.notifyOnVideoPublishedAfterScheduledUpdate(v)
+ if (!video) {
+ return { video, published: false }
}
+
+ await addVideoJobsAfterUpdate({ video, oldPrivacy, isNewVideo, nameChanged: false })
+
+ return { video, published }
}
static get Instance () {