+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 { SCHEDULER_INTERVALS_MS } from '../../initializers/constants'
+import { sequelizeTypescript } from '../../initializers/database'
import { ScheduleVideoUpdateModel } from '../../models/video/schedule-video-update'
-import { retryTransactionWrapper } from '../../helpers/database-utils'
-import { federateVideoIfNeeded } from '../activitypub'
-import { SCHEDULER_INTERVALS_MS, sequelizeTypescript } from '../../initializers'
-import { VideoPrivacy } from '../../../shared/models/videos'
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
- return sequelizeTypescript.transaction(async t => {
- const schedules = await ScheduleVideoUpdateModel.listVideosToUpdate(t)
+ const schedules = await ScheduleVideoUpdateModel.listVideosToUpdate()
+
+ for (const schedule of schedules) {
+ const videoOnly = await VideoModel.load(schedule.videoId)
+ const mutexReleaser = await VideoPathManager.Instance.lockFiles(videoOnly.uuid)
+
+ try {
+ const { video, published } = await this.updateAVideo(schedule)
- for (const schedule of schedules) {
- const video = schedule.Video
- logger.info('Executing scheduled video update on %s.', video.uuid)
+ if (published) Notifier.Instance.notifyOnVideoPublishedAfterScheduledUpdate(video)
+ } catch (err) {
+ logger.error('Cannot update video', { err })
+ }
+
+ mutexReleaser()
+ }
+ }
- if (schedule.privacy) {
- const oldPrivacy = video.privacy
- const isNewVideo = oldPrivacy === VideoPrivacy.PRIVATE
+ private async updateAVideo (schedule: MScheduleVideoUpdate) {
+ let oldPrivacy: VideoPrivacy
+ let isNewVideo: boolean
+ let published = false
- video.privacy = schedule.privacy
- if (isNewVideo === true) video.publishedAt = new Date()
+ const video = await sequelizeTypescript.transaction(async t => {
+ const video = await VideoModel.loadFull(schedule.videoId, t)
+ if (video.state === VideoState.TO_TRANSCODE) return null
- await video.save({ transaction: t })
- await federateVideoIfNeeded(video, isNewVideo, t)
+ logger.info('Executing scheduled video update on %s.', video.uuid)
- if (oldPrivacy === VideoPrivacy.UNLISTED || oldPrivacy === VideoPrivacy.PRIVATE) {
- Notifier.Instance.notifyOnNewVideo(video)
- }
- }
+ if (schedule.privacy) {
+ isNewVideo = video.isNewVideo(schedule.privacy)
+ oldPrivacy = video.privacy
- await schedule.destroy({ transaction: t })
+ setVideoPrivacy(video, schedule.privacy)
+ await video.save({ transaction: t })
+
+ if (oldPrivacy === VideoPrivacy.PRIVATE) {
+ published = true
+ }
}
+
+ await schedule.destroy({ transaction: t })
+
+ return video
})
+
+ if (!video) {
+ return { video, published: false }
+ }
+
+ await addVideoJobsAfterUpdate({ video, oldPrivacy, isNewVideo, nameChanged: false })
+
+ return { video, published }
}
static get Instance () {