Add create-import-video-file-job command
This commit is contained in:
parent
157b62b1f4
commit
0138af9237
|
@ -40,6 +40,7 @@
|
||||||
"start": "node dist/server",
|
"start": "node dist/server",
|
||||||
"update-host": "node ./dist/scripts/update-host.js",
|
"update-host": "node ./dist/scripts/update-host.js",
|
||||||
"create-transcoding-job": "node ./dist/scripts/create-transcoding-job.js",
|
"create-transcoding-job": "node ./dist/scripts/create-transcoding-job.js",
|
||||||
|
"create-import-video-file-job": "node ./dist/scripts/create-import-video-file-job.js",
|
||||||
"test": "scripty",
|
"test": "scripty",
|
||||||
"help": "scripty",
|
"help": "scripty",
|
||||||
"generate-api-doc": "scripty",
|
"generate-api-doc": "scripty",
|
||||||
|
|
39
scripts/create-import-video-file-job.ts
Normal file
39
scripts/create-import-video-file-job.ts
Normal file
|
@ -0,0 +1,39 @@
|
||||||
|
import * as program from 'commander'
|
||||||
|
import { resolve } from 'path'
|
||||||
|
import { VideoModel } from '../server/models/video/video'
|
||||||
|
import { initDatabaseModels } from '../server/initializers'
|
||||||
|
import { JobQueue } from '../server/lib/job-queue'
|
||||||
|
|
||||||
|
program
|
||||||
|
.option('-v, --video [videoUUID]', 'Video UUID')
|
||||||
|
.option('-i, --import [videoFile]', 'Video file')
|
||||||
|
.description('Import a video file to replace an already uploaded file or to add a new resolution')
|
||||||
|
.parse(process.argv)
|
||||||
|
|
||||||
|
if (program['video'] === undefined || program['import'] === undefined) {
|
||||||
|
console.error('All parameters are mandatory.')
|
||||||
|
process.exit(-1)
|
||||||
|
}
|
||||||
|
|
||||||
|
run()
|
||||||
|
.then(() => process.exit(0))
|
||||||
|
.catch(err => {
|
||||||
|
console.error(err)
|
||||||
|
process.exit(-1)
|
||||||
|
})
|
||||||
|
|
||||||
|
async function run () {
|
||||||
|
await initDatabaseModels(true)
|
||||||
|
|
||||||
|
const video = await VideoModel.loadByUUID(program['video'])
|
||||||
|
if (!video) throw new Error('Video not found.')
|
||||||
|
|
||||||
|
const dataInput = {
|
||||||
|
videoUUID: video.uuid,
|
||||||
|
filePath: resolve(program['import'])
|
||||||
|
}
|
||||||
|
|
||||||
|
await JobQueue.Instance.init()
|
||||||
|
await JobQueue.Instance.createJob({ type: 'video-file-import', payload: dataInput })
|
||||||
|
console.log('Import job for video %s created.', video.uuid)
|
||||||
|
}
|
|
@ -6,7 +6,7 @@
|
||||||
import * as bcrypt from 'bcrypt'
|
import * as bcrypt from 'bcrypt'
|
||||||
import * as createTorrent from 'create-torrent'
|
import * as createTorrent from 'create-torrent'
|
||||||
import { pseudoRandomBytes } from 'crypto'
|
import { pseudoRandomBytes } from 'crypto'
|
||||||
import { readdir, readFile, rename, stat, Stats, unlink, writeFile } from 'fs'
|
import { copyFile, readdir, readFile, rename, stat, Stats, unlink, writeFile } from 'fs'
|
||||||
import * as mkdirp from 'mkdirp'
|
import * as mkdirp from 'mkdirp'
|
||||||
import { isAbsolute, join } from 'path'
|
import { isAbsolute, join } from 'path'
|
||||||
import * as pem from 'pem'
|
import * as pem from 'pem'
|
||||||
|
@ -136,6 +136,7 @@ function promisify2WithVoid<T, U> (func: (arg1: T, arg2: U, cb: (err: any) => vo
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const copyFilePromise = promisify2WithVoid<string, string>(copyFile)
|
||||||
const readFileBufferPromise = promisify1<string, Buffer>(readFile)
|
const readFileBufferPromise = promisify1<string, Buffer>(readFile)
|
||||||
const unlinkPromise = promisify1WithVoid<string>(unlink)
|
const unlinkPromise = promisify1WithVoid<string>(unlink)
|
||||||
const renamePromise = promisify2WithVoid<string, string>(rename)
|
const renamePromise = promisify2WithVoid<string, string>(rename)
|
||||||
|
@ -167,6 +168,7 @@ export {
|
||||||
promisify0,
|
promisify0,
|
||||||
promisify1,
|
promisify1,
|
||||||
|
|
||||||
|
copyFilePromise,
|
||||||
readdirPromise,
|
readdirPromise,
|
||||||
readFileBufferPromise,
|
readFileBufferPromise,
|
||||||
unlinkPromise,
|
unlinkPromise,
|
||||||
|
|
|
@ -74,6 +74,7 @@ const JOB_ATTEMPTS: { [ id in JobType ]: number } = {
|
||||||
'activitypub-http-unicast': 5,
|
'activitypub-http-unicast': 5,
|
||||||
'activitypub-http-fetcher': 5,
|
'activitypub-http-fetcher': 5,
|
||||||
'activitypub-follow': 5,
|
'activitypub-follow': 5,
|
||||||
|
'video-file-import': 1,
|
||||||
'video-file': 1,
|
'video-file': 1,
|
||||||
'email': 5
|
'email': 5
|
||||||
}
|
}
|
||||||
|
@ -82,6 +83,7 @@ const JOB_CONCURRENCY: { [ id in JobType ]: number } = {
|
||||||
'activitypub-http-unicast': 5,
|
'activitypub-http-unicast': 5,
|
||||||
'activitypub-http-fetcher': 1,
|
'activitypub-http-fetcher': 1,
|
||||||
'activitypub-follow': 3,
|
'activitypub-follow': 3,
|
||||||
|
'video-file-import': 1,
|
||||||
'video-file': 1,
|
'video-file': 1,
|
||||||
'email': 5
|
'email': 5
|
||||||
}
|
}
|
||||||
|
|
|
@ -16,6 +16,28 @@ export type VideoFilePayload = {
|
||||||
isPortraitMode?: boolean
|
isPortraitMode?: boolean
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export type VideoImportPayload = {
|
||||||
|
videoUUID: string,
|
||||||
|
filePath: string
|
||||||
|
}
|
||||||
|
|
||||||
|
async function processVideoImport (job: kue.Job) {
|
||||||
|
const payload = job.data as VideoImportPayload
|
||||||
|
logger.info('Processing video import in job %d.', job.id)
|
||||||
|
|
||||||
|
const video = await VideoModel.loadByUUIDAndPopulateAccountAndServerAndTags(payload.videoUUID)
|
||||||
|
// No video, maybe deleted?
|
||||||
|
if (!video) {
|
||||||
|
logger.info('Do not process job %d, video does not exist.', job.id, { videoUUID: video.uuid })
|
||||||
|
return undefined
|
||||||
|
}
|
||||||
|
|
||||||
|
await video.importVideoFile(payload.filePath)
|
||||||
|
|
||||||
|
await onVideoFileTranscoderOrImportSuccess(video)
|
||||||
|
return video
|
||||||
|
}
|
||||||
|
|
||||||
async function processVideoFile (job: kue.Job) {
|
async function processVideoFile (job: kue.Job) {
|
||||||
const payload = job.data as VideoFilePayload
|
const payload = job.data as VideoFilePayload
|
||||||
logger.info('Processing video file in job %d.', job.id)
|
logger.info('Processing video file in job %d.', job.id)
|
||||||
|
@ -30,7 +52,7 @@ async function processVideoFile (job: kue.Job) {
|
||||||
// Transcoding in other resolution
|
// Transcoding in other resolution
|
||||||
if (payload.resolution) {
|
if (payload.resolution) {
|
||||||
await video.transcodeOriginalVideofile(payload.resolution, payload.isPortraitMode)
|
await video.transcodeOriginalVideofile(payload.resolution, payload.isPortraitMode)
|
||||||
await onVideoFileTranscoderSuccess(video)
|
await onVideoFileTranscoderOrImportSuccess(video)
|
||||||
} else {
|
} else {
|
||||||
await video.optimizeOriginalVideofile()
|
await video.optimizeOriginalVideofile()
|
||||||
await onVideoFileOptimizerSuccess(video, payload.isNewVideo)
|
await onVideoFileOptimizerSuccess(video, payload.isNewVideo)
|
||||||
|
@ -39,7 +61,7 @@ async function processVideoFile (job: kue.Job) {
|
||||||
return video
|
return video
|
||||||
}
|
}
|
||||||
|
|
||||||
async function onVideoFileTranscoderSuccess (video: VideoModel) {
|
async function onVideoFileTranscoderOrImportSuccess (video: VideoModel) {
|
||||||
if (video === undefined) return undefined
|
if (video === undefined) return undefined
|
||||||
|
|
||||||
// Maybe the video changed in database, refresh it
|
// Maybe the video changed in database, refresh it
|
||||||
|
@ -109,5 +131,6 @@ async function onVideoFileOptimizerSuccess (video: VideoModel, isNewVideo: boole
|
||||||
// ---------------------------------------------------------------------------
|
// ---------------------------------------------------------------------------
|
||||||
|
|
||||||
export {
|
export {
|
||||||
processVideoFile
|
processVideoFile,
|
||||||
|
processVideoImport
|
||||||
}
|
}
|
||||||
|
|
|
@ -7,7 +7,7 @@ import { ActivitypubHttpBroadcastPayload, processActivityPubHttpBroadcast } from
|
||||||
import { ActivitypubHttpFetcherPayload, processActivityPubHttpFetcher } from './handlers/activitypub-http-fetcher'
|
import { ActivitypubHttpFetcherPayload, processActivityPubHttpFetcher } from './handlers/activitypub-http-fetcher'
|
||||||
import { ActivitypubHttpUnicastPayload, processActivityPubHttpUnicast } from './handlers/activitypub-http-unicast'
|
import { ActivitypubHttpUnicastPayload, processActivityPubHttpUnicast } from './handlers/activitypub-http-unicast'
|
||||||
import { EmailPayload, processEmail } from './handlers/email'
|
import { EmailPayload, processEmail } from './handlers/email'
|
||||||
import { processVideoFile, VideoFilePayload } from './handlers/video-file'
|
import { processVideoFile, processVideoImport, VideoFilePayload, VideoImportPayload } from './handlers/video-file'
|
||||||
import { ActivitypubFollowPayload, processActivityPubFollow } from './handlers/activitypub-follow'
|
import { ActivitypubFollowPayload, processActivityPubFollow } from './handlers/activitypub-follow'
|
||||||
|
|
||||||
type CreateJobArgument =
|
type CreateJobArgument =
|
||||||
|
@ -15,6 +15,7 @@ type CreateJobArgument =
|
||||||
{ type: 'activitypub-http-unicast', payload: ActivitypubHttpUnicastPayload } |
|
{ type: 'activitypub-http-unicast', payload: ActivitypubHttpUnicastPayload } |
|
||||||
{ type: 'activitypub-http-fetcher', payload: ActivitypubHttpFetcherPayload } |
|
{ type: 'activitypub-http-fetcher', payload: ActivitypubHttpFetcherPayload } |
|
||||||
{ type: 'activitypub-follow', payload: ActivitypubFollowPayload } |
|
{ type: 'activitypub-follow', payload: ActivitypubFollowPayload } |
|
||||||
|
{ type: 'video-file-import', payload: VideoImportPayload } |
|
||||||
{ type: 'video-file', payload: VideoFilePayload } |
|
{ type: 'video-file', payload: VideoFilePayload } |
|
||||||
{ type: 'email', payload: EmailPayload }
|
{ type: 'email', payload: EmailPayload }
|
||||||
|
|
||||||
|
@ -23,6 +24,7 @@ const handlers: { [ id in JobType ]: (job: kue.Job) => Promise<any>} = {
|
||||||
'activitypub-http-unicast': processActivityPubHttpUnicast,
|
'activitypub-http-unicast': processActivityPubHttpUnicast,
|
||||||
'activitypub-http-fetcher': processActivityPubHttpFetcher,
|
'activitypub-http-fetcher': processActivityPubHttpFetcher,
|
||||||
'activitypub-follow': processActivityPubFollow,
|
'activitypub-follow': processActivityPubFollow,
|
||||||
|
'video-file-import': processVideoImport,
|
||||||
'video-file': processVideoFile,
|
'video-file': processVideoFile,
|
||||||
'email': processEmail
|
'email': processEmail
|
||||||
}
|
}
|
||||||
|
|
|
@ -2,7 +2,7 @@ import * as Bluebird from 'bluebird'
|
||||||
import { map, maxBy } from 'lodash'
|
import { map, maxBy } from 'lodash'
|
||||||
import * as magnetUtil from 'magnet-uri'
|
import * as magnetUtil from 'magnet-uri'
|
||||||
import * as parseTorrent from 'parse-torrent'
|
import * as parseTorrent from 'parse-torrent'
|
||||||
import { join } from 'path'
|
import { join, extname } from 'path'
|
||||||
import * as Sequelize from 'sequelize'
|
import * as Sequelize from 'sequelize'
|
||||||
import {
|
import {
|
||||||
AllowNull,
|
AllowNull,
|
||||||
|
@ -32,6 +32,7 @@ import { VideoFilter } from '../../../shared/models/videos/video-query.type'
|
||||||
import {
|
import {
|
||||||
createTorrentPromise,
|
createTorrentPromise,
|
||||||
peertubeTruncate,
|
peertubeTruncate,
|
||||||
|
copyFilePromise,
|
||||||
renamePromise,
|
renamePromise,
|
||||||
statPromise,
|
statPromise,
|
||||||
unlinkPromise,
|
unlinkPromise,
|
||||||
|
@ -1315,6 +1316,38 @@ export class VideoModel extends Model<VideoModel> {
|
||||||
this.VideoFiles.push(newVideoFile)
|
this.VideoFiles.push(newVideoFile)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async importVideoFile (inputFilePath: string) {
|
||||||
|
let updatedVideoFile = new VideoFileModel({
|
||||||
|
resolution: (await getVideoFileResolution(inputFilePath)).videoFileResolution,
|
||||||
|
extname: extname(inputFilePath),
|
||||||
|
size: (await statPromise(inputFilePath)).size,
|
||||||
|
videoId: this.id
|
||||||
|
})
|
||||||
|
|
||||||
|
const outputPath = this.getVideoFilePath(updatedVideoFile)
|
||||||
|
await copyFilePromise(inputFilePath, outputPath)
|
||||||
|
|
||||||
|
const currentVideoFile = this.VideoFiles.find(videoFile => videoFile.resolution === updatedVideoFile.resolution)
|
||||||
|
const isNewVideoFile = !currentVideoFile
|
||||||
|
|
||||||
|
if (!isNewVideoFile) {
|
||||||
|
if (currentVideoFile.extname !== updatedVideoFile.extname) {
|
||||||
|
await this.removeFile(currentVideoFile)
|
||||||
|
currentVideoFile.set('extname', updatedVideoFile.extname)
|
||||||
|
}
|
||||||
|
currentVideoFile.set('size', updatedVideoFile.size)
|
||||||
|
updatedVideoFile = currentVideoFile
|
||||||
|
}
|
||||||
|
|
||||||
|
await this.createTorrentAndSetInfoHash(updatedVideoFile)
|
||||||
|
|
||||||
|
await updatedVideoFile.save()
|
||||||
|
|
||||||
|
if (isNewVideoFile) {
|
||||||
|
this.VideoFiles.push(updatedVideoFile)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
getOriginalFileResolution () {
|
getOriginalFileResolution () {
|
||||||
const originalFilePath = this.getVideoFilePath(this.getOriginalFile())
|
const originalFilePath = this.getVideoFilePath(this.getOriginalFile())
|
||||||
|
|
||||||
|
|
BIN
server/tests/api/fixtures/video_short-480.webm
Normal file
BIN
server/tests/api/fixtures/video_short-480.webm
Normal file
Binary file not shown.
111
server/tests/cli/create-import-video-file-job.ts
Normal file
111
server/tests/cli/create-import-video-file-job.ts
Normal file
|
@ -0,0 +1,111 @@
|
||||||
|
/* tslint:disable:no-unused-expression */
|
||||||
|
|
||||||
|
import 'mocha'
|
||||||
|
import * as chai from 'chai'
|
||||||
|
import { VideoDetails, VideoFile } from '../../../shared/models/videos'
|
||||||
|
const expect = chai.expect
|
||||||
|
|
||||||
|
import {
|
||||||
|
execCLI,
|
||||||
|
flushTests,
|
||||||
|
getEnvCli,
|
||||||
|
getVideosList,
|
||||||
|
killallServers,
|
||||||
|
parseTorrentVideo,
|
||||||
|
runServer,
|
||||||
|
ServerInfo,
|
||||||
|
setAccessTokensToServers,
|
||||||
|
uploadVideo,
|
||||||
|
wait,
|
||||||
|
getVideo, flushAndRunMultipleServers, doubleFollow
|
||||||
|
} from '../utils'
|
||||||
|
|
||||||
|
function assertVideoProperties (video: VideoFile, resolution: number, extname: string) {
|
||||||
|
expect(video).to.have.nested.property('resolution.id', resolution)
|
||||||
|
expect(video).to.have.property('magnetUri').that.includes(`.${extname}`)
|
||||||
|
expect(video).to.have.property('torrentUrl').that.includes(`-${resolution}.torrent`)
|
||||||
|
expect(video).to.have.property('fileUrl').that.includes(`.${extname}`)
|
||||||
|
expect(video).to.have.property('size').that.is.above(0)
|
||||||
|
}
|
||||||
|
|
||||||
|
describe('Test create import video jobs', function () {
|
||||||
|
this.timeout(60000)
|
||||||
|
|
||||||
|
let servers: ServerInfo[] = []
|
||||||
|
let video1UUID: string
|
||||||
|
let video2UUID: string
|
||||||
|
|
||||||
|
before(async function () {
|
||||||
|
this.timeout(90000)
|
||||||
|
await flushTests()
|
||||||
|
|
||||||
|
// Run server 2 to have transcoding enabled
|
||||||
|
servers = await flushAndRunMultipleServers(2)
|
||||||
|
await setAccessTokensToServers(servers)
|
||||||
|
|
||||||
|
await doubleFollow(servers[0], servers[1])
|
||||||
|
|
||||||
|
// Upload two videos for our needs
|
||||||
|
const res1 = await uploadVideo(servers[0].url, servers[0].accessToken, { name: 'video1' })
|
||||||
|
video1UUID = res1.body.video.uuid
|
||||||
|
const res2 = await uploadVideo(servers[1].url, servers[1].accessToken, { name: 'video2' })
|
||||||
|
video2UUID = res2.body.video.uuid
|
||||||
|
|
||||||
|
await wait(40000)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should run a import job on video 1 with a lower resolution', async function () {
|
||||||
|
const env = getEnvCli(servers[0])
|
||||||
|
await execCLI(`${env} npm run create-import-video-file-job -- -v ${video1UUID} -i server/tests/api/fixtures/video_short-480.webm`)
|
||||||
|
|
||||||
|
await wait(30000)
|
||||||
|
|
||||||
|
for (const server of servers) {
|
||||||
|
const { data: videos } = (await getVideosList(server.url)).body
|
||||||
|
expect(videos).to.have.lengthOf(2)
|
||||||
|
|
||||||
|
let infoHashes: { [ id: number ]: string } = {}
|
||||||
|
|
||||||
|
const video = videos.find(({ uuid }) => uuid === video1UUID)
|
||||||
|
const videoDetail: VideoDetails = (await getVideo(server.url, video.uuid)).body
|
||||||
|
|
||||||
|
expect(videoDetail.files).to.have.lengthOf(2)
|
||||||
|
const [originalVideo, transcodedVideo] = videoDetail.files
|
||||||
|
assertVideoProperties(originalVideo, 720, 'webm')
|
||||||
|
assertVideoProperties(transcodedVideo, 480, 'webm')
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should run a import job on video 2 with the same resolution', async function () {
|
||||||
|
const env = getEnvCli(servers[1])
|
||||||
|
await execCLI(`${env} npm run create-import-video-file-job -- -v ${video2UUID} -i server/tests/api/fixtures/video_short.ogv`)
|
||||||
|
|
||||||
|
await wait(30000)
|
||||||
|
|
||||||
|
for (const server of servers.reverse()) {
|
||||||
|
const { data: videos } = (await getVideosList(server.url)).body
|
||||||
|
expect(videos).to.have.lengthOf(2)
|
||||||
|
|
||||||
|
let infoHashes: { [ id: number ]: string }
|
||||||
|
|
||||||
|
const video = videos.find(({ uuid }) => uuid === video2UUID)
|
||||||
|
const videoDetail: VideoDetails = (await getVideo(server.url, video.uuid)).body
|
||||||
|
|
||||||
|
expect(videoDetail.files).to.have.lengthOf(4)
|
||||||
|
const [originalVideo, transcodedVideo420, transcodedVideo320, transcodedVideo240] = videoDetail.files
|
||||||
|
assertVideoProperties(originalVideo, 720, 'ogv')
|
||||||
|
assertVideoProperties(transcodedVideo420, 480, 'mp4')
|
||||||
|
assertVideoProperties(transcodedVideo320, 360, 'mp4')
|
||||||
|
assertVideoProperties(transcodedVideo240, 240, 'mp4')
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
after(async function () {
|
||||||
|
killallServers(servers)
|
||||||
|
|
||||||
|
// Keep the logs if the test failed
|
||||||
|
if (this['ok']) {
|
||||||
|
await flushTests()
|
||||||
|
}
|
||||||
|
})
|
||||||
|
})
|
|
@ -1,4 +1,5 @@
|
||||||
// Order of the tests we want to execute
|
// Order of the tests we want to execute
|
||||||
import './create-transcoding-job'
|
import './create-transcoding-job'
|
||||||
|
import './create-import-video-file-job'
|
||||||
import './reset-password'
|
import './reset-password'
|
||||||
import './update-host'
|
import './update-host'
|
||||||
|
|
|
@ -4,6 +4,7 @@ export type JobType = 'activitypub-http-unicast' |
|
||||||
'activitypub-http-broadcast' |
|
'activitypub-http-broadcast' |
|
||||||
'activitypub-http-fetcher' |
|
'activitypub-http-fetcher' |
|
||||||
'activitypub-follow' |
|
'activitypub-follow' |
|
||||||
|
'video-file-import' |
|
||||||
'video-file' |
|
'video-file' |
|
||||||
'email'
|
'email'
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue
Block a user