Set bitrate limits for transcoding (fixes #638) (#1135)

* Set bitrate limits for transcoding (fixes #638)

* added optimization script and test, changed stuff

* fix test, improve docs

* re-add optimize-old-videos script

* added documentation

* Don't optimize videos without valid UUID, or redundancy videos

* move getUUIDFromFilename

* fix tests?

* update torrent and file size, some more fixes/improvements

* use higher bitrate for high fps video, adjust bitrates

* add test video

* don't throw error if resolution is undefined

* generate test fixture on the fly

* use random noise video for bitrate test, add promise

* shorten test video to avoid timeout

* use existing function to optimize video

* various fixes

* increase test timeout

* limit test fixture size, add link

* test fixes

* add await

* more test fixes, add -b:v parameter

* replace ffmpeg wiki link

* fix ffmpeg params

* fix unit test

* add test fixture to .gitgnore

* add video transcoding fps model

* add missing file
This commit is contained in:
Felix Ableitner 2018-10-08 09:26:04 -05:00 committed by Chocobozzz
parent 2cae5f1307
commit edb4ffc7e0
16 changed files with 221 additions and 26 deletions

1
.gitignore vendored
View File

@ -9,6 +9,7 @@
/test4/ /test4/
/test5/ /test5/
/test6/ /test6/
/server/tests/fixtures/video_high_bitrate_1080p.mp4
# Production # Production
/storage/ /storage/

View File

@ -51,6 +51,7 @@
"generate-api-doc": "scripty", "generate-api-doc": "scripty",
"parse-log": "node ./dist/scripts/parse-log.js", "parse-log": "node ./dist/scripts/parse-log.js",
"prune-storage": "node ./dist/scripts/prune-storage.js", "prune-storage": "node ./dist/scripts/prune-storage.js",
"optimize-old-videos": "node ./dist/scripts/optimize-old-videos.js",
"postinstall": "cd client && yarn install --pure-lockfile", "postinstall": "cd client && yarn install --pure-lockfile",
"tsc": "tsc", "tsc": "tsc",
"spectacle-docs": "node_modules/spectacle-docs/bin/spectacle.js", "spectacle-docs": "node_modules/spectacle-docs/bin/spectacle.js",

View File

@ -18,6 +18,7 @@ printf " reset-password -- -u [user] -> Reset the password of user [user]\n"
printf " create-transcoding-job -- -v [video UUID] \n" printf " create-transcoding-job -- -v [video UUID] \n"
printf " -> Create a transcoding job for a particular video\n" printf " -> Create a transcoding job for a particular video\n"
printf " prune-storage -> Delete (after confirmation) unknown video files/thumbnails/previews... (due to a bad video deletion, transcoding job not finished...)\n" printf " prune-storage -> Delete (after confirmation) unknown video files/thumbnails/previews... (due to a bad video deletion, transcoding job not finished...)\n"
printf " optimize-old-videos -> Re-transcode videos that have a high bitrate, to make them suitable for streaming over slow connections"
printf " dev -> Watch, run the livereload and run the server so that you can develop the application\n" printf " dev -> Watch, run the livereload and run the server so that you can develop the application\n"
printf " start -> Run the server\n" printf " start -> Run the server\n"
printf " update-host -> Upgrade scheme/host in torrent files according to the webserver configuration (config/ folder)\n" printf " update-host -> Upgrade scheme/host in torrent files according to the webserver configuration (config/ folder)\n"

View File

@ -0,0 +1,36 @@
import { join } from 'path'
import { readdir } from 'fs-extra'
import { CONFIG, VIDEO_TRANSCODING_FPS } from '../server/initializers/constants'
import { getVideoFileResolution, getVideoFileBitrate, getVideoFileFPS } from '../server/helpers/ffmpeg-utils'
import { getMaxBitrate } from '../shared/models/videos'
import { VideoRedundancyModel } from '../server/models/redundancy/video-redundancy'
import { VideoModel } from '../server/models/video/video'
import { getUUIDFromFilename } from '../server/helpers/utils'
import { optimizeVideofile } from '../server/lib/video-transcoding'
run()
.then(() => process.exit(0))
.catch(err => {
console.error(err)
process.exit(-1)
})
async function run () {
const files = await readdir(CONFIG.STORAGE.VIDEOS_DIR)
for (const file of files) {
const inputPath = join(CONFIG.STORAGE.VIDEOS_DIR, file)
const videoBitrate = await getVideoFileBitrate(inputPath)
const fps = await getVideoFileFPS(inputPath)
const resolution = await getVideoFileResolution(inputPath)
const uuid = getUUIDFromFilename(file)
const isLocalVideo = await VideoRedundancyModel.isLocalByVideoUUIDExists(uuid)
const isMaxBitrateExceeded =
videoBitrate > getMaxBitrate(resolution.videoFileResolution, fps, VIDEO_TRANSCODING_FPS)
if (uuid && isLocalVideo && isMaxBitrateExceeded) {
const videoModel = await VideoModel.loadByUUIDWithFile(uuid)
await optimizeVideofile(videoModel, inputPath)
}
}
console.log('Finished optimizing videos')
}

View File

@ -5,6 +5,7 @@ import { VideoModel } from '../server/models/video/video'
import { initDatabaseModels } from '../server/initializers' import { initDatabaseModels } from '../server/initializers'
import { remove, readdir } from 'fs-extra' import { remove, readdir } from 'fs-extra'
import { VideoRedundancyModel } from '../server/models/redundancy/video-redundancy' import { VideoRedundancyModel } from '../server/models/redundancy/video-redundancy'
import { getUUIDFromFilename } from '../server/helpers/utils'
run() run()
.then(() => process.exit(0)) .then(() => process.exit(0))
@ -82,15 +83,6 @@ async function pruneDirectory (directory: string, onlyOwned = false) {
return toDelete return toDelete
} }
function getUUIDFromFilename (filename: string) {
const regex = /[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{12}/
const result = filename.match(regex)
if (!result || Array.isArray(result) === false) return null
return result[0]
}
async function askConfirmation () { async function askConfirmation () {
return new Promise((res, rej) => { return new Promise((res, rej) => {
prompt.start() prompt.start()

View File

@ -1,6 +1,6 @@
import * as ffmpeg from 'fluent-ffmpeg' import * as ffmpeg from 'fluent-ffmpeg'
import { join } from 'path' import { join } from 'path'
import { VideoResolution } from '../../shared/models/videos' import { VideoResolution, getTargetBitrate } from '../../shared/models/videos'
import { CONFIG, FFMPEG_NICE, VIDEO_TRANSCODING_FPS } from '../initializers' import { CONFIG, FFMPEG_NICE, VIDEO_TRANSCODING_FPS } from '../initializers'
import { processImage } from './image-utils' import { processImage } from './image-utils'
import { logger } from './logger' import { logger } from './logger'
@ -55,6 +55,16 @@ async function getVideoFileFPS (path: string) {
return 0 return 0
} }
async function getVideoFileBitrate (path: string) {
return new Promise<number>((res, rej) => {
ffmpeg.ffprobe(path, (err, metadata) => {
if (err) return rej(err)
return res(metadata.format.bit_rate)
})
})
}
function getDurationFromVideoFile (path: string) { function getDurationFromVideoFile (path: string) {
return new Promise<number>((res, rej) => { return new Promise<number>((res, rej) => {
ffmpeg.ffprobe(path, (err, metadata) => { ffmpeg.ffprobe(path, (err, metadata) => {
@ -138,6 +148,12 @@ function transcode (options: TranscodeOptions) {
command = command.withFPS(fps) command = command.withFPS(fps)
} }
// Constrained Encoding (VBV)
// https://slhck.info/video/2017/03/01/rate-control.html
// https://trac.ffmpeg.org/wiki/Limiting%20the%20output%20bitrate
const targetBitrate = getTargetBitrate(options.resolution, fps, VIDEO_TRANSCODING_FPS)
command.outputOptions([`-b:v ${ targetBitrate }`, `-maxrate ${ targetBitrate }`, `-bufsize ${ targetBitrate * 2 }`])
command command
.on('error', (err, stdout, stderr) => { .on('error', (err, stdout, stderr) => {
logger.error('Error in transcoding job.', { stdout, stderr }) logger.error('Error in transcoding job.', { stdout, stderr })
@ -157,7 +173,8 @@ export {
transcode, transcode,
getVideoFileFPS, getVideoFileFPS,
computeResolutionsToTranscode, computeResolutionsToTranscode,
audio audio,
getVideoFileBitrate
} }
// --------------------------------------------------------------------------- // ---------------------------------------------------------------------------

View File

@ -77,6 +77,20 @@ async function getVersion () {
return require('../../../package.json').version return require('../../../package.json').version
} }
/**
* From a filename like "ede4cba5-742b-46fa-a388-9a6eb3a3aeb3.mp4", returns
* only the "ede4cba5-742b-46fa-a388-9a6eb3a3aeb3" part. If the filename does
* not contain a UUID, returns null.
*/
function getUUIDFromFilename (filename: string) {
const regex = /[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{12}/
const result = filename.match(regex)
if (!result || Array.isArray(result) === false) return null
return result[0]
}
// --------------------------------------------------------------------------- // ---------------------------------------------------------------------------
export { export {
@ -86,5 +100,6 @@ export {
getSecureTorrentName, getSecureTorrentName,
getServerActor, getServerActor,
getVersion, getVersion,
generateVideoTmpPath generateVideoTmpPath,
getUUIDFromFilename
} }

View File

@ -3,7 +3,7 @@ import { dirname, join } from 'path'
import { JobType, VideoRateType, VideoState, VideosRedundancy } from '../../shared/models' import { JobType, VideoRateType, VideoState, VideosRedundancy } from '../../shared/models'
import { ActivityPubActorType } from '../../shared/models/activitypub' import { ActivityPubActorType } from '../../shared/models/activitypub'
import { FollowState } from '../../shared/models/actors' import { FollowState } from '../../shared/models/actors'
import { VideoAbuseState, VideoImportState, VideoPrivacy } from '../../shared/models/videos' import { VideoAbuseState, VideoImportState, VideoPrivacy, VideoTranscodingFPS } from '../../shared/models/videos'
// Do not use barrels, remain constants as independent as possible // Do not use barrels, remain constants as independent as possible
import { buildPath, isTestInstance, parseDuration, root, sanitizeHost, sanitizeUrl } from '../helpers/core-utils' import { buildPath, isTestInstance, parseDuration, root, sanitizeHost, sanitizeUrl } from '../helpers/core-utils'
import { NSFWPolicyType } from '../../shared/models/videos/nsfw-policy.type' import { NSFWPolicyType } from '../../shared/models/videos/nsfw-policy.type'
@ -393,7 +393,7 @@ const RATES_LIMIT = {
} }
let VIDEO_VIEW_LIFETIME = 60000 * 60 // 1 hour let VIDEO_VIEW_LIFETIME = 60000 * 60 // 1 hour
const VIDEO_TRANSCODING_FPS = { const VIDEO_TRANSCODING_FPS: VideoTranscodingFPS = {
MIN: 10, MIN: 10,
AVERAGE: 30, AVERAGE: 30,
MAX: 60, MAX: 60,

View File

@ -1,7 +1,7 @@
import { ACTIVITY_PUB, JOB_REQUEST_TIMEOUT } from '../../initializers' import { ACTIVITY_PUB, JOB_REQUEST_TIMEOUT } from '../../initializers'
import { doRequest } from '../../helpers/requests' import { doRequest } from '../../helpers/requests'
import { logger } from '../../helpers/logger' import { logger } from '../../helpers/logger'
import Bluebird = require('bluebird') import * as Bluebird from 'bluebird'
async function crawlCollectionPage <T> (uri: string, handler: (items: T[]) => Promise<any> | Bluebird<any>) { async function crawlCollectionPage <T> (uri: string, handler: (items: T[]) => Promise<any> | Bluebird<any>) {
logger.info('Crawling ActivityPub data on %s.', uri) logger.info('Crawling ActivityPub data on %s.', uri)

View File

@ -8,7 +8,7 @@ import { retryTransactionWrapper } from '../../../helpers/database-utils'
import { sequelizeTypescript } from '../../../initializers' import { sequelizeTypescript } from '../../../initializers'
import * as Bluebird from 'bluebird' import * as Bluebird from 'bluebird'
import { computeResolutionsToTranscode } from '../../../helpers/ffmpeg-utils' import { computeResolutionsToTranscode } from '../../../helpers/ffmpeg-utils'
import { importVideoFile, transcodeOriginalVideofile, optimizeOriginalVideofile } from '../../video-transcoding' import { importVideoFile, transcodeOriginalVideofile, optimizeVideofile } from '../../video-transcoding'
export type VideoFilePayload = { export type VideoFilePayload = {
videoUUID: string videoUUID: string
@ -56,7 +56,7 @@ async function processVideoFile (job: Bull.Job) {
await retryTransactionWrapper(onVideoFileTranscoderOrImportSuccess, video) await retryTransactionWrapper(onVideoFileTranscoderOrImportSuccess, video)
} else { } else {
await optimizeOriginalVideofile(video) await optimizeVideofile(video)
await retryTransactionWrapper(onVideoFileOptimizerSuccess, video, payload.isNewVideo) await retryTransactionWrapper(onVideoFileOptimizerSuccess, video, payload.isNewVideo)
} }

View File

@ -1,5 +1,5 @@
import { CONFIG } from '../initializers' import { CONFIG } from '../initializers'
import { join, extname } from 'path' import { join, extname, basename } from 'path'
import { getVideoFileFPS, getVideoFileResolution, transcode } from '../helpers/ffmpeg-utils' import { getVideoFileFPS, getVideoFileResolution, transcode } from '../helpers/ffmpeg-utils'
import { copy, remove, rename, stat } from 'fs-extra' import { copy, remove, rename, stat } from 'fs-extra'
import { logger } from '../helpers/logger' import { logger } from '../helpers/logger'
@ -7,11 +7,16 @@ import { VideoResolution } from '../../shared/models/videos'
import { VideoFileModel } from '../models/video/video-file' import { VideoFileModel } from '../models/video/video-file'
import { VideoModel } from '../models/video/video' import { VideoModel } from '../models/video/video'
async function optimizeOriginalVideofile (video: VideoModel) { async function optimizeVideofile (video: VideoModel, videoInputPath?: string) {
const videosDirectory = CONFIG.STORAGE.VIDEOS_DIR const videosDirectory = CONFIG.STORAGE.VIDEOS_DIR
const newExtname = '.mp4' const newExtname = '.mp4'
const inputVideoFile = video.getOriginalFile() let inputVideoFile = null
const videoInputPath = join(videosDirectory, video.getVideoFilename(inputVideoFile)) if (videoInputPath == null) {
inputVideoFile = video.getOriginalFile()
videoInputPath = join(videosDirectory, video.getVideoFilename(inputVideoFile))
} else {
inputVideoFile = basename(videoInputPath)
}
const videoTranscodedPath = join(videosDirectory, video.id + '-transcoded' + newExtname) const videoTranscodedPath = join(videosDirectory, video.id + '-transcoded' + newExtname)
const transcodeOptions = { const transcodeOptions = {
@ -124,7 +129,7 @@ async function importVideoFile (video: VideoModel, inputFilePath: string) {
} }
export { export {
optimizeOriginalVideofile, optimizeVideofile,
transcodeOriginalVideofile, transcodeOriginalVideofile,
importVideoFile importVideoFile
} }

View File

@ -4,8 +4,8 @@ import * as chai from 'chai'
import 'mocha' import 'mocha'
import { omit } from 'lodash' import { omit } from 'lodash'
import * as ffmpeg from 'fluent-ffmpeg' import * as ffmpeg from 'fluent-ffmpeg'
import { VideoDetails, VideoState } from '../../../../shared/models/videos' import { VideoDetails, VideoState, getMaxBitrate, VideoResolution } from '../../../../shared/models/videos'
import { getVideoFileFPS, audio } from '../../../helpers/ffmpeg-utils' import { getVideoFileFPS, audio, getVideoFileBitrate, getVideoFileResolution } from '../../../helpers/ffmpeg-utils'
import { import {
buildAbsoluteFixturePath, buildAbsoluteFixturePath,
doubleFollow, doubleFollow,
@ -20,8 +20,10 @@ import {
uploadVideo, uploadVideo,
webtorrentAdd webtorrentAdd
} from '../../utils' } from '../../utils'
import { join } from 'path' import { join, basename } from 'path'
import { waitJobs } from '../../utils/server/jobs' import { waitJobs } from '../../utils/server/jobs'
import { remove } from 'fs-extra'
import { VIDEO_TRANSCODING_FPS } from '../../../../server/initializers/constants'
const expect = chai.expect const expect = chai.expect
@ -228,7 +230,7 @@ describe('Test video transcoding', function () {
} }
}) })
it('Should wait transcoding before publishing the video', async function () { it('Should wait for transcoding before publishing the video', async function () {
this.timeout(80000) this.timeout(80000)
{ {
@ -281,7 +283,59 @@ describe('Test video transcoding', function () {
} }
}) })
const tempFixturePath = buildAbsoluteFixturePath('video_high_bitrate_1080p.mp4')
it('Should respect maximum bitrate values', async function () {
this.timeout(160000)
{
// Generate a random, high bitrate video on the fly, so we don't have to include
// a large file in the repo. The video needs to have a certain minimum length so
// that FFmpeg properly applies bitrate limits.
// https://stackoverflow.com/a/15795112
await new Promise<void>(async (res, rej) => {
ffmpeg()
.outputOptions(['-f rawvideo', '-video_size 1920x1080', '-i /dev/urandom'])
.outputOptions(['-ac 2', '-f s16le', '-i /dev/urandom', '-t 10'])
.outputOptions(['-maxrate 10M', '-bufsize 10M'])
.output(tempFixturePath)
.on('error', rej)
.on('end', res)
.run()
})
const bitrate = await getVideoFileBitrate(tempFixturePath)
expect(bitrate).to.be.above(getMaxBitrate(VideoResolution.H_1080P, 60, VIDEO_TRANSCODING_FPS))
const videoAttributes = {
name: 'high bitrate video',
description: 'high bitrate video',
fixture: basename(tempFixturePath)
}
await uploadVideo(servers[1].url, servers[1].accessToken, videoAttributes)
await waitJobs(servers)
for (const server of servers) {
const res = await getVideosList(server.url)
const video = res.body.data.find(v => v.name === videoAttributes.name)
for (const resolution of ['240', '360', '480', '720', '1080']) {
const path = join(root(), 'test2', 'videos', video.uuid + '-' + resolution + '.mp4')
const bitrate = await getVideoFileBitrate(path)
const fps = await getVideoFileFPS(path)
const resolution2 = await getVideoFileResolution(path)
expect(resolution2.videoFileResolution.toString()).to.equal(resolution)
expect(bitrate).to.be.below(getMaxBitrate(resolution2.videoFileResolution, fps, VIDEO_TRANSCODING_FPS))
}
}
}
})
after(async function () { after(async function () {
remove(tempFixturePath)
killallServers(servers) killallServers(servers)
}) })
}) })

View File

@ -21,6 +21,7 @@ export * from './video-update.model'
export * from './video.model' export * from './video.model'
export * from './video-query.type' export * from './video-query.type'
export * from './video-state.enum' export * from './video-state.enum'
export * from './video-transcoding-fps.model'
export * from './caption/video-caption.model' export * from './caption/video-caption.model'
export * from './caption/video-caption-update.model' export * from './caption/video-caption-update.model'
export * from './import/video-import-create.model' export * from './import/video-import-create.model'

View File

@ -1,3 +1,5 @@
import { VideoTranscodingFPS } from './video-transcoding-fps.model'
export enum VideoResolution { export enum VideoResolution {
H_240P = 240, H_240P = 240,
H_360P = 360, H_360P = 360,
@ -5,3 +7,56 @@ export enum VideoResolution {
H_720P = 720, H_720P = 720,
H_1080P = 1080 H_1080P = 1080
} }
/**
* Bitrate targets for different resolutions and frame rates, in bytes per second.
* Sources for individual quality levels:
* Google Live Encoder: https://support.google.com/youtube/answer/2853702?hl=en
* YouTube Video Info (tested with random music video): https://www.h3xed.com/blogmedia/youtube-info.php
*/
export function getTargetBitrate (resolution: VideoResolution, fps: number,
fpsTranscodingConstants: VideoTranscodingFPS) {
switch (resolution) {
case VideoResolution.H_240P:
// quality according to Google Live Encoder: 300 - 700 Kbps
// Quality according to YouTube Video Info: 186 Kbps
return 250 * 1000
case VideoResolution.H_360P:
// quality according to Google Live Encoder: 400 - 1,000 Kbps
// Quality according to YouTube Video Info: 480 Kbps
return 500 * 1000
case VideoResolution.H_480P:
// quality according to Google Live Encoder: 500 - 2,000 Kbps
// Quality according to YouTube Video Info: 879 Kbps
return 900 * 1000
case VideoResolution.H_720P:
if (fps === fpsTranscodingConstants.MAX) {
// quality according to Google Live Encoder: 2,250 - 6,000 Kbps
// Quality according to YouTube Video Info: 2634 Kbps
return 2600 * 1000
} else {
// quality according to Google Live Encoder: 1,500 - 4,000 Kbps
// Quality according to YouTube Video Info: 1752 Kbps
return 1750 * 1000
}
case VideoResolution.H_1080P: // fallthrough
default:
if (fps === fpsTranscodingConstants.MAX) {
// quality according to Google Live Encoder: 3000 - 6000 Kbps
// Quality according to YouTube Video Info: 4387 Kbps
return 4400 * 1000
} else {
// quality according to Google Live Encoder: 3000 - 6000 Kbps
// Quality according to YouTube Video Info: 3277 Kbps
return 3300 * 1000
}
}
}
/**
* The maximum bitrate we expect to see on a transcoded video in bytes per second.
*/
export function getMaxBitrate (resolution: VideoResolution, fps: number,
fpsTranscodingConstants: VideoTranscodingFPS) {
return getTargetBitrate(resolution, fps, fpsTranscodingConstants) * 2
}

View File

@ -0,0 +1,6 @@
export type VideoTranscodingFPS = {
MIN: number,
AVERAGE: number,
MAX: number,
KEEP_ORIGIN_FPS_RESOLUTION_MIN: number
}

View File

@ -187,6 +187,17 @@ To delete them (a confirmation will be demanded first):
$ sudo -u peertube NODE_CONFIG_DIR=/var/www/peertube/config NODE_ENV=production npm run prune-storage $ sudo -u peertube NODE_CONFIG_DIR=/var/www/peertube/config NODE_ENV=production npm run prune-storage
``` ```
### optimize-old-videos.js
Before version v1.0.0-beta.16, Peertube did not specify a bitrate for the transcoding of uploaded videos.
This means that videos might be encoded into very large files that are too large for streaming. This script
re-transcodes these videos so that they can be watched properly, even on slow connections.
```
$ sudo -u peertube NODE_CONFIG_DIR=/var/www/peertube/config NODE_ENV=production npm run optimize-old-videos
```
### update-host.js ### update-host.js
If you started PeerTube with a domain, and then changed it you will have invalid torrent files and invalid URLs in your database. If you started PeerTube with a domain, and then changed it you will have invalid torrent files and invalid URLs in your database.