2024-03-15 09:47:18 -05:00
|
|
|
import { buildAspectRatio } from '@peertube/peertube-core-utils'
|
2024-07-23 09:38:51 -05:00
|
|
|
import {
|
|
|
|
MergeAudioTranscodeOptions,
|
|
|
|
TranscodeVODOptionsType,
|
|
|
|
VideoTranscodeOptions,
|
|
|
|
getVideoStreamDuration
|
|
|
|
} from '@peertube/peertube-ffmpeg'
|
|
|
|
import { VideoFileStream } from '@peertube/peertube-models'
|
2023-07-31 07:34:36 -05:00
|
|
|
import { computeOutputFPS } from '@server/helpers/ffmpeg/index.js'
|
|
|
|
import { createTorrentAndSetInfoHash } from '@server/helpers/webtorrent.js'
|
|
|
|
import { VideoModel } from '@server/models/video/video.js'
|
|
|
|
import { MVideoFile, MVideoFullLight } from '@server/types/models/index.js'
|
2024-03-15 09:47:18 -05:00
|
|
|
import { Job } from 'bullmq'
|
|
|
|
import { move, remove } from 'fs-extra/esm'
|
|
|
|
import { copyFile } from 'fs/promises'
|
|
|
|
import { basename, join } from 'path'
|
2023-07-31 07:34:36 -05:00
|
|
|
import { CONFIG } from '../../initializers/config.js'
|
|
|
|
import { VideoFileModel } from '../../models/video/video-file.js'
|
|
|
|
import { JobQueue } from '../job-queue/index.js'
|
|
|
|
import { generateWebVideoFilename } from '../paths.js'
|
2024-03-15 09:47:18 -05:00
|
|
|
import { buildNewFile, saveNewOriginalFileIfNeeded } from '../video-file.js'
|
|
|
|
import { buildStoryboardJobIfNeeded } from '../video-jobs.js'
|
2023-07-31 07:34:36 -05:00
|
|
|
import { VideoPathManager } from '../video-path-manager.js'
|
|
|
|
import { buildFFmpegVOD } from './shared/index.js'
|
|
|
|
import { buildOriginalFileResolution } from './transcoding-resolutions.js'
|
2023-04-21 07:55:10 -05:00
|
|
|
|
|
|
|
// Optimize the original video file and replace it. The resolution is not changed.
|
|
|
|
export async function optimizeOriginalVideofile (options: {
|
|
|
|
video: MVideoFullLight
|
|
|
|
quickTranscode: boolean
|
|
|
|
job: Job
|
|
|
|
}) {
|
2024-07-23 09:38:51 -05:00
|
|
|
const { quickTranscode, job } = options
|
2023-04-21 07:55:10 -05:00
|
|
|
|
|
|
|
const transcodeDirectory = CONFIG.STORAGE.TMP_DIR
|
|
|
|
const newExtname = '.mp4'
|
|
|
|
|
|
|
|
// Will be released by our transcodeVOD function once ffmpeg is ran
|
2024-07-23 09:38:51 -05:00
|
|
|
const inputFileMutexReleaser = await VideoPathManager.Instance.lockFiles(options.video.uuid)
|
2023-04-21 07:55:10 -05:00
|
|
|
|
|
|
|
try {
|
2024-07-23 09:38:51 -05:00
|
|
|
const video = await VideoModel.loadFull(options.video.id)
|
|
|
|
const inputVideoFile = video.getMaxQualityFile(VideoFileStream.VIDEO)
|
2023-04-21 07:55:10 -05:00
|
|
|
|
2024-07-23 09:38:51 -05:00
|
|
|
const result = await VideoPathManager.Instance.makeAvailableVideoFile(inputVideoFile, async videoInputPath => {
|
2023-04-21 07:55:10 -05:00
|
|
|
const videoOutputPath = join(transcodeDirectory, video.id + '-transcoded' + newExtname)
|
|
|
|
|
|
|
|
const transcodeType: TranscodeVODOptionsType = quickTranscode
|
|
|
|
? 'quick-transcode'
|
|
|
|
: 'video'
|
|
|
|
|
|
|
|
const resolution = buildOriginalFileResolution(inputVideoFile.resolution)
|
2024-08-12 09:17:11 -05:00
|
|
|
const fps = computeOutputFPS({ inputFPS: inputVideoFile.fps, resolution, isOriginResolution: true, type: 'vod' })
|
2023-04-21 07:55:10 -05:00
|
|
|
|
|
|
|
// Could be very long!
|
|
|
|
await buildFFmpegVOD(job).transcode({
|
|
|
|
type: transcodeType,
|
|
|
|
|
2024-07-23 09:38:51 -05:00
|
|
|
videoInputPath,
|
2023-04-21 07:55:10 -05:00
|
|
|
outputPath: videoOutputPath,
|
|
|
|
|
|
|
|
inputFileMutexReleaser,
|
|
|
|
|
|
|
|
resolution,
|
|
|
|
fps
|
|
|
|
})
|
|
|
|
|
2024-02-27 04:18:56 -06:00
|
|
|
const { videoFile } = await onWebVideoFileTranscoding({ video, videoOutputPath, deleteWebInputVideoFile: inputVideoFile })
|
2023-04-21 07:55:10 -05:00
|
|
|
|
|
|
|
return { transcodeType, videoFile }
|
|
|
|
})
|
|
|
|
|
|
|
|
return result
|
|
|
|
} finally {
|
|
|
|
inputFileMutexReleaser()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-03-15 09:47:18 -05:00
|
|
|
// Transcode the original/old/source video file to a lower resolution compatible with web browsers
|
2023-07-11 02:21:13 -05:00
|
|
|
export async function transcodeNewWebVideoResolution (options: {
|
2023-04-21 07:55:10 -05:00
|
|
|
video: MVideoFullLight
|
2023-07-31 07:34:36 -05:00
|
|
|
resolution: number
|
2023-04-21 07:55:10 -05:00
|
|
|
fps: number
|
|
|
|
job: Job
|
|
|
|
}) {
|
2023-04-21 09:31:04 -05:00
|
|
|
const { video: videoArg, resolution, fps, job } = options
|
2023-04-21 07:55:10 -05:00
|
|
|
|
|
|
|
const transcodeDirectory = CONFIG.STORAGE.TMP_DIR
|
|
|
|
const newExtname = '.mp4'
|
|
|
|
|
2023-04-21 09:31:04 -05:00
|
|
|
const inputFileMutexReleaser = await VideoPathManager.Instance.lockFiles(videoArg.uuid)
|
2023-04-21 07:55:10 -05:00
|
|
|
|
|
|
|
try {
|
2023-04-21 09:31:04 -05:00
|
|
|
const video = await VideoModel.loadFull(videoArg.uuid)
|
2023-04-21 07:55:10 -05:00
|
|
|
|
2024-07-23 09:38:51 -05:00
|
|
|
const result = await VideoPathManager.Instance.makeAvailableMaxQualityFiles(video, async ({ videoPath, separatedAudioPath }) => {
|
2024-02-27 04:18:56 -06:00
|
|
|
const filename = generateWebVideoFilename(resolution, newExtname)
|
|
|
|
const videoOutputPath = join(transcodeDirectory, filename)
|
2023-04-21 07:55:10 -05:00
|
|
|
|
2024-07-23 09:38:51 -05:00
|
|
|
const transcodeOptions: VideoTranscodeOptions = {
|
|
|
|
type: 'video',
|
|
|
|
|
|
|
|
videoInputPath: videoPath,
|
|
|
|
separatedAudioInputPath: separatedAudioPath,
|
2023-04-21 07:55:10 -05:00
|
|
|
|
|
|
|
outputPath: videoOutputPath,
|
|
|
|
|
|
|
|
inputFileMutexReleaser,
|
|
|
|
|
|
|
|
resolution,
|
|
|
|
fps
|
|
|
|
}
|
|
|
|
|
|
|
|
await buildFFmpegVOD(job).transcode(transcodeOptions)
|
|
|
|
|
2024-02-27 04:18:56 -06:00
|
|
|
return onWebVideoFileTranscoding({ video, videoOutputPath })
|
2023-04-21 07:55:10 -05:00
|
|
|
})
|
|
|
|
|
|
|
|
return result
|
|
|
|
} finally {
|
|
|
|
inputFileMutexReleaser()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Merge an image with an audio file to create a video
|
|
|
|
export async function mergeAudioVideofile (options: {
|
|
|
|
video: MVideoFullLight
|
2023-07-31 07:34:36 -05:00
|
|
|
resolution: number
|
2023-04-21 07:55:10 -05:00
|
|
|
fps: number
|
|
|
|
job: Job
|
|
|
|
}) {
|
2023-04-21 09:31:04 -05:00
|
|
|
const { video: videoArg, resolution, fps, job } = options
|
2023-04-21 07:55:10 -05:00
|
|
|
|
|
|
|
const transcodeDirectory = CONFIG.STORAGE.TMP_DIR
|
|
|
|
const newExtname = '.mp4'
|
|
|
|
|
2023-04-21 09:31:04 -05:00
|
|
|
const inputFileMutexReleaser = await VideoPathManager.Instance.lockFiles(videoArg.uuid)
|
2023-04-21 07:55:10 -05:00
|
|
|
|
|
|
|
try {
|
2023-04-21 09:31:04 -05:00
|
|
|
const video = await VideoModel.loadFull(videoArg.uuid)
|
2024-07-23 09:38:51 -05:00
|
|
|
const inputVideoFile = video.getMaxQualityFile(VideoFileStream.AUDIO)
|
2023-04-21 07:55:10 -05:00
|
|
|
|
2024-07-23 09:38:51 -05:00
|
|
|
const result = await VideoPathManager.Instance.makeAvailableVideoFile(inputVideoFile, async audioInputPath => {
|
2023-04-21 07:55:10 -05:00
|
|
|
const videoOutputPath = join(transcodeDirectory, video.id + '-transcoded' + newExtname)
|
|
|
|
|
|
|
|
// If the user updates the video preview during transcoding
|
|
|
|
const previewPath = video.getPreview().getPath()
|
|
|
|
const tmpPreviewPath = join(CONFIG.STORAGE.TMP_DIR, basename(previewPath))
|
|
|
|
await copyFile(previewPath, tmpPreviewPath)
|
|
|
|
|
2024-07-23 09:38:51 -05:00
|
|
|
const transcodeOptions: MergeAudioTranscodeOptions = {
|
|
|
|
type: 'merge-audio',
|
|
|
|
|
|
|
|
videoInputPath: tmpPreviewPath,
|
|
|
|
audioPath: audioInputPath,
|
2023-04-21 07:55:10 -05:00
|
|
|
|
|
|
|
outputPath: videoOutputPath,
|
|
|
|
|
|
|
|
inputFileMutexReleaser,
|
|
|
|
|
|
|
|
resolution,
|
|
|
|
fps
|
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
|
|
|
await buildFFmpegVOD(job).transcode(transcodeOptions)
|
|
|
|
|
|
|
|
await remove(tmpPreviewPath)
|
|
|
|
} catch (err) {
|
|
|
|
await remove(tmpPreviewPath)
|
|
|
|
throw err
|
|
|
|
}
|
|
|
|
|
2024-02-27 04:18:56 -06:00
|
|
|
await onWebVideoFileTranscoding({
|
2023-04-21 07:55:10 -05:00
|
|
|
video,
|
2023-06-01 07:51:16 -05:00
|
|
|
videoOutputPath,
|
2024-02-27 04:18:56 -06:00
|
|
|
deleteWebInputVideoFile: inputVideoFile,
|
2023-06-01 07:51:16 -05:00
|
|
|
wasAudioFile: true
|
2023-04-21 07:55:10 -05:00
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
return result
|
|
|
|
} finally {
|
|
|
|
inputFileMutexReleaser()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-07-11 02:21:13 -05:00
|
|
|
export async function onWebVideoFileTranscoding (options: {
|
2023-04-21 07:55:10 -05:00
|
|
|
video: MVideoFullLight
|
|
|
|
videoOutputPath: string
|
2023-06-01 07:51:16 -05:00
|
|
|
wasAudioFile?: boolean // default false
|
2024-02-27 04:18:56 -06:00
|
|
|
deleteWebInputVideoFile?: MVideoFile
|
2023-04-21 07:55:10 -05:00
|
|
|
}) {
|
2024-02-27 04:18:56 -06:00
|
|
|
const { video, videoOutputPath, wasAudioFile, deleteWebInputVideoFile } = options
|
2023-04-21 07:55:10 -05:00
|
|
|
|
|
|
|
const mutexReleaser = await VideoPathManager.Instance.lockFiles(video.uuid)
|
|
|
|
|
2024-02-27 04:18:56 -06:00
|
|
|
const videoFile = await buildNewFile({ mode: 'web-video', path: videoOutputPath })
|
|
|
|
videoFile.videoId = video.id
|
|
|
|
|
2023-04-21 07:55:10 -05:00
|
|
|
try {
|
|
|
|
await video.reload()
|
|
|
|
|
2024-02-27 04:18:56 -06:00
|
|
|
// ffmpeg generated a new video file, so update the video duration
|
|
|
|
// See https://trac.ffmpeg.org/ticket/5456
|
|
|
|
if (wasAudioFile) {
|
|
|
|
video.duration = await getVideoStreamDuration(videoOutputPath)
|
|
|
|
video.aspectRatio = buildAspectRatio({ width: videoFile.width, height: videoFile.height })
|
|
|
|
await video.save()
|
|
|
|
}
|
2023-04-21 07:55:10 -05:00
|
|
|
|
2024-02-27 04:18:56 -06:00
|
|
|
const outputPath = VideoPathManager.Instance.getFSVideoFileOutputPath(video, videoFile)
|
2023-04-21 07:55:10 -05:00
|
|
|
|
|
|
|
await move(videoOutputPath, outputPath, { overwrite: true })
|
|
|
|
|
|
|
|
await createTorrentAndSetInfoHash(video, videoFile)
|
|
|
|
|
2024-02-27 04:18:56 -06:00
|
|
|
if (deleteWebInputVideoFile) {
|
2024-03-15 09:47:18 -05:00
|
|
|
await saveNewOriginalFileIfNeeded(video, deleteWebInputVideoFile)
|
|
|
|
|
2024-02-27 04:18:56 -06:00
|
|
|
await video.removeWebVideoFile(deleteWebInputVideoFile)
|
|
|
|
await deleteWebInputVideoFile.destroy()
|
|
|
|
}
|
|
|
|
|
2024-03-15 09:47:18 -05:00
|
|
|
const existingFile = await VideoFileModel.loadWebVideoFile({ videoId: video.id, fps: videoFile.fps, resolution: videoFile.resolution })
|
|
|
|
if (existingFile) await video.removeWebVideoFile(existingFile)
|
|
|
|
|
2023-04-21 07:55:10 -05:00
|
|
|
await VideoFileModel.customUpsert(videoFile, 'video', undefined)
|
|
|
|
video.VideoFiles = await video.$get('VideoFiles')
|
|
|
|
|
2023-06-01 07:51:16 -05:00
|
|
|
if (wasAudioFile) {
|
2023-12-27 03:39:09 -06:00
|
|
|
await JobQueue.Instance.createJob(buildStoryboardJobIfNeeded({ video, federate: false }))
|
2023-06-01 07:51:16 -05:00
|
|
|
}
|
|
|
|
|
2023-04-21 07:55:10 -05:00
|
|
|
return { video, videoFile }
|
|
|
|
} finally {
|
|
|
|
mutexReleaser()
|
|
|
|
}
|
|
|
|
}
|