]> git.immae.eu Git - github/Chocobozzz/PeerTube.git/blob - server/lib/video-transcoding.ts
Merge branch 'feature/correctly-send-activities' into develop
[github/Chocobozzz/PeerTube.git] / server / lib / video-transcoding.ts
1 import { CONFIG, HLS_PLAYLIST_DIRECTORY } from '../initializers'
2 import { extname, join } from 'path'
3 import { getVideoFileFPS, getVideoFileResolution, transcode } from '../helpers/ffmpeg-utils'
4 import { copy, ensureDir, move, remove, stat } from 'fs-extra'
5 import { logger } from '../helpers/logger'
6 import { VideoResolution } from '../../shared/models/videos'
7 import { VideoFileModel } from '../models/video/video-file'
8 import { VideoModel } from '../models/video/video'
9 import { updateMasterHLSPlaylist, updateSha256Segments } from './hls'
10 import { VideoStreamingPlaylistModel } from '../models/video/video-streaming-playlist'
11 import { VideoStreamingPlaylistType } from '../../shared/models/videos/video-streaming-playlist.type'
12
13 async function optimizeVideofile (video: VideoModel, inputVideoFileArg?: VideoFileModel) {
14 const videosDirectory = CONFIG.STORAGE.VIDEOS_DIR
15 const newExtname = '.mp4'
16
17 const inputVideoFile = inputVideoFileArg ? inputVideoFileArg : video.getOriginalFile()
18 const videoInputPath = join(videosDirectory, video.getVideoFilename(inputVideoFile))
19 const videoTranscodedPath = join(videosDirectory, video.id + '-transcoded' + newExtname)
20
21 const transcodeOptions = {
22 inputPath: videoInputPath,
23 outputPath: videoTranscodedPath,
24 resolution: inputVideoFile.resolution
25 }
26
27 // Could be very long!
28 await transcode(transcodeOptions)
29
30 try {
31 await remove(videoInputPath)
32
33 // Important to do this before getVideoFilename() to take in account the new file extension
34 inputVideoFile.set('extname', newExtname)
35
36 const videoOutputPath = video.getVideoFilePath(inputVideoFile)
37 await move(videoTranscodedPath, videoOutputPath)
38 const stats = await stat(videoOutputPath)
39 const fps = await getVideoFileFPS(videoOutputPath)
40
41 inputVideoFile.set('size', stats.size)
42 inputVideoFile.set('fps', fps)
43
44 await video.createTorrentAndSetInfoHash(inputVideoFile)
45 await inputVideoFile.save()
46 } catch (err) {
47 // Auto destruction...
48 video.destroy().catch(err => logger.error('Cannot destruct video after transcoding failure.', { err }))
49
50 throw err
51 }
52 }
53
54 async function transcodeOriginalVideofile (video: VideoModel, resolution: VideoResolution, isPortrait: boolean) {
55 const videosDirectory = CONFIG.STORAGE.VIDEOS_DIR
56 const extname = '.mp4'
57
58 // We are sure it's x264 in mp4 because optimizeOriginalVideofile was already executed
59 const videoInputPath = join(videosDirectory, video.getVideoFilename(video.getOriginalFile()))
60
61 const newVideoFile = new VideoFileModel({
62 resolution,
63 extname,
64 size: 0,
65 videoId: video.id
66 })
67 const videoOutputPath = join(CONFIG.STORAGE.VIDEOS_DIR, video.getVideoFilename(newVideoFile))
68
69 const transcodeOptions = {
70 inputPath: videoInputPath,
71 outputPath: videoOutputPath,
72 resolution,
73 isPortraitMode: isPortrait
74 }
75
76 await transcode(transcodeOptions)
77
78 const stats = await stat(videoOutputPath)
79 const fps = await getVideoFileFPS(videoOutputPath)
80
81 newVideoFile.set('size', stats.size)
82 newVideoFile.set('fps', fps)
83
84 await video.createTorrentAndSetInfoHash(newVideoFile)
85
86 await newVideoFile.save()
87
88 video.VideoFiles.push(newVideoFile)
89 }
90
91 async function generateHlsPlaylist (video: VideoModel, resolution: VideoResolution, isPortraitMode: boolean) {
92 const baseHlsDirectory = join(HLS_PLAYLIST_DIRECTORY, video.uuid)
93 await ensureDir(join(HLS_PLAYLIST_DIRECTORY, video.uuid))
94
95 const videoInputPath = join(CONFIG.STORAGE.VIDEOS_DIR, video.getVideoFilename(video.getOriginalFile()))
96 const outputPath = join(baseHlsDirectory, VideoStreamingPlaylistModel.getHlsPlaylistFilename(resolution))
97
98 const transcodeOptions = {
99 inputPath: videoInputPath,
100 outputPath,
101 resolution,
102 isPortraitMode,
103
104 hlsPlaylist: {
105 videoFilename: VideoStreamingPlaylistModel.getHlsVideoName(video.uuid, resolution)
106 }
107 }
108
109 await transcode(transcodeOptions)
110
111 await updateMasterHLSPlaylist(video)
112 await updateSha256Segments(video)
113
114 const playlistUrl = CONFIG.WEBSERVER.URL + VideoStreamingPlaylistModel.getHlsMasterPlaylistStaticPath(video.uuid)
115
116 await VideoStreamingPlaylistModel.upsert({
117 videoId: video.id,
118 playlistUrl,
119 segmentsSha256Url: CONFIG.WEBSERVER.URL + VideoStreamingPlaylistModel.getHlsSha256SegmentsStaticPath(video.uuid),
120 p2pMediaLoaderInfohashes: VideoStreamingPlaylistModel.buildP2PMediaLoaderInfoHashes(playlistUrl, video.VideoFiles),
121
122 type: VideoStreamingPlaylistType.HLS
123 })
124 }
125
126 async function importVideoFile (video: VideoModel, inputFilePath: string) {
127 const { videoFileResolution } = await getVideoFileResolution(inputFilePath)
128 const { size } = await stat(inputFilePath)
129 const fps = await getVideoFileFPS(inputFilePath)
130
131 let updatedVideoFile = new VideoFileModel({
132 resolution: videoFileResolution,
133 extname: extname(inputFilePath),
134 size,
135 fps,
136 videoId: video.id
137 })
138
139 const currentVideoFile = video.VideoFiles.find(videoFile => videoFile.resolution === updatedVideoFile.resolution)
140
141 if (currentVideoFile) {
142 // Remove old file and old torrent
143 await video.removeFile(currentVideoFile)
144 await video.removeTorrent(currentVideoFile)
145 // Remove the old video file from the array
146 video.VideoFiles = video.VideoFiles.filter(f => f !== currentVideoFile)
147
148 // Update the database
149 currentVideoFile.set('extname', updatedVideoFile.extname)
150 currentVideoFile.set('size', updatedVideoFile.size)
151 currentVideoFile.set('fps', updatedVideoFile.fps)
152
153 updatedVideoFile = currentVideoFile
154 }
155
156 const outputPath = video.getVideoFilePath(updatedVideoFile)
157 await copy(inputFilePath, outputPath)
158
159 await video.createTorrentAndSetInfoHash(updatedVideoFile)
160
161 await updatedVideoFile.save()
162
163 video.VideoFiles.push(updatedVideoFile)
164 }
165
166 export {
167 generateHlsPlaylist,
168 optimizeVideofile,
169 transcodeOriginalVideofile,
170 importVideoFile
171 }