]> git.immae.eu Git - github/Chocobozzz/PeerTube.git/blob - server/lib/schedulers/update-videos-scheduler.ts
Add test on AP hooks
[github/Chocobozzz/PeerTube.git] / server / lib / schedulers / update-videos-scheduler.ts
1 import { VideoModel } from '@server/models/video/video'
2 import { MScheduleVideoUpdate } from '@server/types/models'
3 import { VideoPrivacy, VideoState } from '@shared/models'
4 import { logger } from '../../helpers/logger'
5 import { SCHEDULER_INTERVALS_MS } from '../../initializers/constants'
6 import { sequelizeTypescript } from '../../initializers/database'
7 import { ScheduleVideoUpdateModel } from '../../models/video/schedule-video-update'
8 import { Notifier } from '../notifier'
9 import { addVideoJobsAfterUpdate } from '../video'
10 import { VideoPathManager } from '../video-path-manager'
11 import { setVideoPrivacy } from '../video-privacy'
12 import { AbstractScheduler } from './abstract-scheduler'
13
14 export class UpdateVideosScheduler extends AbstractScheduler {
15
16 private static instance: AbstractScheduler
17
18 protected schedulerIntervalMs = SCHEDULER_INTERVALS_MS.UPDATE_VIDEOS
19
20 private constructor () {
21 super()
22 }
23
24 protected async internalExecute () {
25 return this.updateVideos()
26 }
27
28 private async updateVideos () {
29 if (!await ScheduleVideoUpdateModel.areVideosToUpdate()) return undefined
30
31 const schedules = await ScheduleVideoUpdateModel.listVideosToUpdate()
32
33 for (const schedule of schedules) {
34 const videoOnly = await VideoModel.load(schedule.videoId)
35 const mutexReleaser = await VideoPathManager.Instance.lockFiles(videoOnly.uuid)
36
37 try {
38 const { video, published } = await this.updateAVideo(schedule)
39
40 if (published) Notifier.Instance.notifyOnVideoPublishedAfterScheduledUpdate(video)
41 } catch (err) {
42 logger.error('Cannot update video', { err })
43 }
44
45 mutexReleaser()
46 }
47 }
48
49 private async updateAVideo (schedule: MScheduleVideoUpdate) {
50 let oldPrivacy: VideoPrivacy
51 let isNewVideo: boolean
52 let published = false
53
54 const video = await sequelizeTypescript.transaction(async t => {
55 const video = await VideoModel.loadFull(schedule.videoId, t)
56 if (video.state === VideoState.TO_TRANSCODE) return null
57
58 logger.info('Executing scheduled video update on %s.', video.uuid)
59
60 if (schedule.privacy) {
61 isNewVideo = video.isNewVideo(schedule.privacy)
62 oldPrivacy = video.privacy
63
64 setVideoPrivacy(video, schedule.privacy)
65 await video.save({ transaction: t })
66
67 if (oldPrivacy === VideoPrivacy.PRIVATE) {
68 published = true
69 }
70 }
71
72 await schedule.destroy({ transaction: t })
73
74 return video
75 })
76
77 if (!video) {
78 return { video, published: false }
79 }
80
81 await addVideoJobsAfterUpdate({ video, oldPrivacy, isNewVideo, nameChanged: false })
82
83 return { video, published }
84 }
85
86 static get Instance () {
87 return this.instance || (this.instance = new this())
88 }
89 }