PeerTube/server/core/lib/job-queue/handlers/video-live-ending.ts

374 lines
13 KiB
TypeScript
Raw Normal View History

2024-06-13 02:23:12 -05:00
import { ffprobePromise, getAudioStream, getVideoStreamDimensionsInfo, getVideoStreamFPS } from '@peertube/peertube-ffmpeg'
import { ThumbnailType, VideoFileStream, VideoLiveEndingPayload, VideoState } from '@peertube/peertube-models'
import { peertubeTruncate } from '@server/helpers/core-utils.js'
import { CONSTRAINTS_FIELDS } from '@server/initializers/constants.js'
import { getLocalVideoActivityPubUrl } from '@server/lib/activitypub/url.js'
import { federateVideoIfNeeded } from '@server/lib/activitypub/videos/index.js'
import { cleanupAndDestroyPermanentLive, cleanupTMPLiveFiles, cleanupUnsavedNormalLive } from '@server/lib/live/index.js'
2023-09-01 09:47:25 -05:00
import {
generateHLSMasterPlaylistFilename,
generateHlsSha256SegmentsFilename,
getHLSDirectory,
getLiveReplayBaseDirectory
} from '@server/lib/paths.js'
import { generateLocalVideoMiniature, regenerateMiniaturesIfNeeded, updateLocalVideoMiniatureFromExisting } from '@server/lib/thumbnail.js'
import { generateHlsPlaylistResolutionFromTS } from '@server/lib/transcoding/hls-transcoding.js'
2024-06-13 02:23:12 -05:00
import { createTranscriptionTaskIfNeeded } from '@server/lib/video-captions.js'
import { buildStoryboardJobIfNeeded } from '@server/lib/video-jobs.js'
import { VideoPathManager } from '@server/lib/video-path-manager.js'
2024-06-13 02:23:12 -05:00
import { isVideoInPublicDirectory } from '@server/lib/video-privacy.js'
import { moveToNextState } from '@server/lib/video-state.js'
import { setVideoTags } from '@server/lib/video.js'
import { VideoBlacklistModel } from '@server/models/video/video-blacklist.js'
import { VideoFileModel } from '@server/models/video/video-file.js'
import { VideoLiveReplaySettingModel } from '@server/models/video/video-live-replay-setting.js'
import { VideoLiveSessionModel } from '@server/models/video/video-live-session.js'
import { VideoLiveModel } from '@server/models/video/video-live.js'
import { VideoStreamingPlaylistModel } from '@server/models/video/video-streaming-playlist.js'
import { VideoModel } from '@server/models/video/video.js'
import {
MThumbnail,
MVideo,
MVideoLive,
MVideoLiveSession,
MVideoTag,
MVideoThumbnail,
MVideoWithAllFiles,
MVideoWithFileThumbnail
} from '@server/types/models/index.js'
2024-06-13 02:23:12 -05:00
import { Job } from 'bullmq'
import { remove } from 'fs-extra/esm'
import { readdir } from 'fs/promises'
import { join } from 'path'
import { logger, loggerTagsFactory } from '../../../helpers/logger.js'
import { JobQueue } from '../job-queue.js'
const lTags = loggerTagsFactory('live', 'job')
2021-08-27 07:32:44 -05:00
async function processVideoLiveEnding (job: Job) {
const payload = job.data as VideoLiveEndingPayload
logger.info('Processing video live ending for %s.', payload.videoId, { payload, ...lTags() })
2020-11-04 07:16:57 -06:00
function logError () {
logger.warn('Video live %d does not exist anymore. Cannot process live ending.', payload.videoId, lTags())
2020-11-04 07:16:57 -06:00
}
const video = await VideoModel.load(payload.videoId)
2020-10-26 10:44:23 -05:00
const live = await VideoLiveModel.loadByVideoId(payload.videoId)
2022-05-03 04:38:07 -05:00
const liveSession = await VideoLiveSessionModel.load(payload.liveSessionId)
2020-10-26 10:44:23 -05:00
if (!video || !live || !liveSession) {
2020-11-04 07:16:57 -06:00
logError()
return
}
2022-10-04 04:17:37 -05:00
const permanentLive = live.permanentLive
liveSession.endingProcessed = true
await liveSession.save()
if (liveSession.saveReplay !== true) {
return cleanupLiveAndFederate({ permanentLive, video, streamingPlaylistId: payload.streamingPlaylistId })
2020-10-26 10:44:23 -05:00
}
if (await hasReplayFiles(payload.replayDirectory) !== true) {
logger.info(`No replay files found for live ${video.uuid}, skipping video replay creation.`, { ...lTags(video.uuid) })
return cleanupLiveAndFederate({ permanentLive, video, streamingPlaylistId: payload.streamingPlaylistId })
}
if (permanentLive) {
await saveReplayToExternalVideo({
liveVideo: video,
liveSession,
publishedAt: payload.publishedAt,
replayDirectory: payload.replayDirectory
})
return cleanupLiveAndFederate({ permanentLive, video, streamingPlaylistId: payload.streamingPlaylistId })
}
return replaceLiveByReplay({
video,
liveSession,
live,
permanentLive,
replayDirectory: payload.replayDirectory
})
2020-10-26 10:44:23 -05:00
}
// ---------------------------------------------------------------------------
export {
2021-06-16 08:14:41 -05:00
processVideoLiveEnding
2020-10-26 10:44:23 -05:00
}
// ---------------------------------------------------------------------------
2022-05-03 04:38:07 -05:00
async function saveReplayToExternalVideo (options: {
liveVideo: MVideoThumbnail
2022-05-03 04:38:07 -05:00
liveSession: MVideoLiveSession
publishedAt: string
replayDirectory: string
}) {
const { liveSession, publishedAt, replayDirectory } = options
2022-05-03 04:38:07 -05:00
const liveVideo = await VideoModel.loadFull(options.liveVideo.id)
const replaySettings = await VideoLiveReplaySettingModel.load(liveSession.replaySettingId)
const videoNameSuffix = ` - ${new Date(publishedAt).toLocaleString()}`
const truncatedVideoName = peertubeTruncate(liveVideo.name, {
length: CONSTRAINTS_FIELDS.VIDEOS.NAME.max - videoNameSuffix.length
})
const replayVideo = new VideoModel({
name: truncatedVideoName + videoNameSuffix,
isLive: false,
state: VideoState.TO_TRANSCODE,
duration: 0,
remote: liveVideo.remote,
category: liveVideo.category,
licence: liveVideo.licence,
language: liveVideo.language,
commentsPolicy: liveVideo.commentsPolicy,
downloadEnabled: liveVideo.downloadEnabled,
2022-05-03 04:38:07 -05:00
waitTranscoding: true,
nsfw: liveVideo.nsfw,
description: liveVideo.description,
2024-02-27 04:18:56 -06:00
aspectRatio: liveVideo.aspectRatio,
support: liveVideo.support,
privacy: replaySettings.privacy,
channelId: liveVideo.channelId
}) as MVideoWithAllFiles & MVideoTag
replayVideo.Thumbnails = []
replayVideo.VideoFiles = []
replayVideo.VideoStreamingPlaylists = []
replayVideo.url = getLocalVideoActivityPubUrl(replayVideo)
await replayVideo.save()
await setVideoTags({ video: replayVideo, tags: liveVideo.Tags.map(t => t.name) })
liveSession.replayVideoId = replayVideo.id
2022-05-03 04:38:07 -05:00
await liveSession.save()
// If live is blacklisted, also blacklist the replay
const blacklist = await VideoBlacklistModel.loadByVideoId(liveVideo.id)
if (blacklist) {
await VideoBlacklistModel.create({
videoId: replayVideo.id,
unfederated: blacklist.unfederated,
reason: blacklist.reason,
type: blacklist.type
})
}
2023-09-01 09:47:25 -05:00
const inputFileMutexReleaser = await VideoPathManager.Instance.lockFiles(liveVideo.uuid)
2023-09-01 09:47:25 -05:00
try {
await assignReplayFilesToVideo({ video: replayVideo, replayDirectory })
logger.info(`Removing replay directory ${replayDirectory}`, lTags(liveVideo.uuid))
2023-09-01 09:47:25 -05:00
await remove(replayDirectory)
} finally {
inputFileMutexReleaser()
}
try {
await copyOrRegenerateThumbnails({ liveVideo, replayVideo })
} catch (err) {
logger.error(
`Cannot copy/regenerate thumbnails of ended live ${liveVideo.uuid} to external video ${replayVideo.uuid}`,
lTags(liveVideo.uuid, replayVideo.uuid)
)
}
2023-06-01 07:51:16 -05:00
await createStoryboardJob(replayVideo)
2024-07-11 04:29:46 -05:00
await createTranscriptionTaskIfNeeded(replayVideo)
2024-06-13 02:23:12 -05:00
await moveToNextState({ video: replayVideo, isNewVideo: true })
}
async function copyOrRegenerateThumbnails (options: {
liveVideo: MVideoThumbnail
replayVideo: MVideoWithFileThumbnail
}) {
const { liveVideo, replayVideo } = options
let thumbnails: MThumbnail[] = []
const preview = liveVideo.getPreview()
if (preview?.automaticallyGenerated === false) {
thumbnails = await Promise.all(
[ ThumbnailType.MINIATURE, ThumbnailType.PREVIEW ].map(type => {
return updateLocalVideoMiniatureFromExisting({
inputPath: preview.getPath(),
video: replayVideo,
type,
2024-10-22 03:30:06 -05:00
automaticallyGenerated: false,
keepOriginal: true
})
})
)
} else {
thumbnails = await generateLocalVideoMiniature({
video: replayVideo,
videoFile: replayVideo.getMaxQualityFile(VideoFileStream.VIDEO) || replayVideo.getMaxQualityFile(VideoFileStream.AUDIO),
types: [ ThumbnailType.MINIATURE, ThumbnailType.PREVIEW ],
ffprobe: undefined
})
}
for (const thumbnail of thumbnails) {
await replayVideo.addAndSaveThumbnail(thumbnail)
}
}
2022-05-03 04:38:07 -05:00
async function replaceLiveByReplay (options: {
video: MVideo
2022-05-03 04:38:07 -05:00
liveSession: MVideoLiveSession
live: MVideoLive
permanentLive: boolean
2022-05-03 04:38:07 -05:00
replayDirectory: string
}) {
2023-09-01 09:47:25 -05:00
const { video: liveVideo, liveSession, live, permanentLive, replayDirectory } = options
2022-05-03 04:38:07 -05:00
const replaySettings = await VideoLiveReplaySettingModel.load(liveSession.replaySettingId)
2023-09-01 09:47:25 -05:00
const videoWithFiles = await VideoModel.loadFull(liveVideo.id)
const hlsPlaylist = videoWithFiles.getHLSPlaylist()
2023-09-01 09:47:25 -05:00
const replayInAnotherDirectory = isVideoInPublicDirectory(liveVideo.privacy) !== isVideoInPublicDirectory(replaySettings.privacy)
logger.info(`Replacing live ${liveVideo.uuid} by replay ${replayDirectory}.`, { replayInAnotherDirectory, ...lTags(liveVideo.uuid) })
await cleanupTMPLiveFiles(videoWithFiles, hlsPlaylist)
2020-10-26 10:44:23 -05:00
2020-10-27 10:06:24 -05:00
await live.destroy()
videoWithFiles.isLive = false
videoWithFiles.privacy = replaySettings.privacy
videoWithFiles.waitTranscoding = true
videoWithFiles.state = VideoState.TO_TRANSCODE
2020-10-28 04:49:20 -05:00
await videoWithFiles.save()
2022-05-03 04:38:07 -05:00
liveSession.replayVideoId = videoWithFiles.id
2022-05-03 04:38:07 -05:00
await liveSession.save()
2020-10-26 10:44:23 -05:00
2024-03-25 09:14:56 -05:00
await VideoFileModel.removeHLSFilesOfStreamingPlaylistId(hlsPlaylist.id)
2021-07-23 04:20:00 -05:00
// Reset playlist
2020-11-03 08:33:30 -06:00
hlsPlaylist.VideoFiles = []
2021-07-23 04:20:00 -05:00
hlsPlaylist.playlistFilename = generateHLSMasterPlaylistFilename()
hlsPlaylist.segmentsSha256Filename = generateHlsSha256SegmentsFilename()
await hlsPlaylist.save()
2020-11-03 08:33:30 -06:00
2023-09-01 09:47:25 -05:00
const inputFileMutexReleaser = await VideoPathManager.Instance.lockFiles(videoWithFiles.uuid)
2023-09-01 09:47:25 -05:00
try {
await assignReplayFilesToVideo({ video: videoWithFiles, replayDirectory })
// Should not happen in this function, but we keep the code if in the future we can replace the permanent live by a replay
if (permanentLive) { // Remove session replay
await remove(replayDirectory)
} else {
// We won't stream again in this live, we can delete the base replay directory
await remove(getLiveReplayBaseDirectory(liveVideo))
// If the live was in another base directory, also delete it
if (replayInAnotherDirectory) {
await remove(getHLSDirectory(liveVideo))
}
}
} finally {
inputFileMutexReleaser()
}
// Regenerate the thumbnail & preview?
try {
await regenerateMiniaturesIfNeeded(videoWithFiles, undefined)
} catch (err) {
logger.error(`Cannot regenerate thumbnails of ended live ${videoWithFiles.uuid}`, lTags(liveVideo.uuid))
}
2022-05-03 04:38:07 -05:00
// We consider this is a new video
await moveToNextState({ video: videoWithFiles, isNewVideo: true })
2023-06-01 07:51:16 -05:00
await createStoryboardJob(videoWithFiles)
2024-07-11 04:29:46 -05:00
await createTranscriptionTaskIfNeeded(videoWithFiles)
}
2022-05-03 04:38:07 -05:00
async function assignReplayFilesToVideo (options: {
video: MVideo
replayDirectory: string
}) {
const { video, replayDirectory } = options
const concatenatedTsFiles = await readdir(replayDirectory)
2023-09-01 09:47:25 -05:00
logger.info(`Assigning replays ${replayDirectory} to video ${video.uuid}.`, { concatenatedTsFiles, ...lTags(video.uuid) })
for (const concatenatedTsFile of concatenatedTsFiles) {
2023-09-01 09:47:25 -05:00
// Generating hls playlist can be long, reload the video in this case
await video.reload()
2020-12-04 08:10:13 -06:00
const concatenatedTsFilePath = join(replayDirectory, concatenatedTsFile)
2020-12-02 03:07:26 -06:00
const probe = await ffprobePromise(concatenatedTsFilePath)
const { audioStream } = await getAudioStream(concatenatedTsFilePath, probe)
const { resolution } = await getVideoStreamDimensionsInfo(concatenatedTsFilePath, probe)
const fps = await getVideoStreamFPS(concatenatedTsFilePath, probe)
2020-12-02 03:07:26 -06:00
try {
await generateHlsPlaylistResolutionFromTS({
video,
2023-09-01 09:47:25 -05:00
inputFileMutexReleaser: null, // Already locked in parent
concatenatedTsFilePath,
resolution,
fps,
isAAC: audioStream?.codec_name === 'aac'
})
} catch (err) {
logger.error('Cannot generate HLS playlist resolution from TS files.', { err })
}
2020-11-03 08:33:30 -06:00
}
2020-10-28 04:49:20 -05:00
return video
}
2020-11-06 03:57:40 -06:00
2022-05-03 04:38:07 -05:00
async function cleanupLiveAndFederate (options: {
video: MVideo
permanentLive: boolean
streamingPlaylistId: number
2022-05-03 04:38:07 -05:00
}) {
const { permanentLive, video, streamingPlaylistId } = options
2020-12-03 07:10:54 -06:00
const streamingPlaylist = await VideoStreamingPlaylistModel.loadWithVideo(streamingPlaylistId)
if (streamingPlaylist) {
if (permanentLive) {
await cleanupAndDestroyPermanentLive(video, streamingPlaylist)
} else {
await cleanupUnsavedNormalLive(video, streamingPlaylist)
}
}
2022-05-25 08:18:29 -05:00
try {
2022-06-28 07:57:51 -05:00
const fullVideo = await VideoModel.loadFull(video.id)
2022-05-25 08:18:29 -05:00
return federateVideoIfNeeded(fullVideo, false, undefined)
} catch (err) {
logger.warn('Cannot federate live after cleanup', { videoId: video.id, err })
}
}
2023-06-01 07:51:16 -05:00
function createStoryboardJob (video: MVideo) {
2023-12-27 03:39:09 -06:00
return JobQueue.Instance.createJob(buildStoryboardJobIfNeeded({ video, federate: true }))
2023-06-01 07:51:16 -05:00
}
async function hasReplayFiles (replayDirectory: string) {
return (await readdir(replayDirectory)).length !== 0
}