2020-01-31 16:56:52 +01:00
|
|
|
// eslint-disable @typescript-eslint/no-unnecessary-type-assertion
|
|
|
|
|
2020-01-28 11:07:23 +01:00
|
|
|
import { registerTSPaths } from '../helpers/register-ts-paths'
|
|
|
|
registerTSPaths()
|
|
|
|
|
|
|
|
import * as program from 'commander'
|
|
|
|
import { getAdminTokenOrDie, getServerCredentials } from './cli'
|
|
|
|
import { VideoRedundanciesTarget, VideoRedundancy } from '@shared/models'
|
|
|
|
import { addVideoRedundancy, listVideoRedundancies, removeVideoRedundancy } from '@shared/extra-utils/server/redundancy'
|
2020-12-08 21:16:10 +01:00
|
|
|
import { HttpStatusCode } from '@shared/core-utils/miscs/http-error-codes'
|
2020-01-28 11:07:23 +01:00
|
|
|
import validator from 'validator'
|
|
|
|
import * as CliTable3 from 'cli-table3'
|
2020-01-31 16:56:52 +01:00
|
|
|
import { URL } from 'url'
|
2020-01-28 11:07:23 +01:00
|
|
|
import { uniq } from 'lodash'
|
|
|
|
|
2020-01-31 16:56:52 +01:00
|
|
|
import bytes = require('bytes')
|
|
|
|
|
2020-01-28 11:07:23 +01:00
|
|
|
program
|
|
|
|
.name('plugins')
|
|
|
|
.usage('[command] [options]')
|
|
|
|
|
|
|
|
program
|
|
|
|
.command('list-remote-redundancies')
|
|
|
|
.description('List remote redundancies on your videos')
|
|
|
|
.option('-u, --url <url>', 'Server url')
|
|
|
|
.option('-U, --username <username>', 'Username')
|
|
|
|
.option('-p, --password <token>', 'Password')
|
|
|
|
.action(() => listRedundanciesCLI('my-videos'))
|
|
|
|
|
|
|
|
program
|
|
|
|
.command('list-my-redundancies')
|
|
|
|
.description('List your redundancies of remote videos')
|
|
|
|
.option('-u, --url <url>', 'Server url')
|
|
|
|
.option('-U, --username <username>', 'Username')
|
|
|
|
.option('-p, --password <token>', 'Password')
|
|
|
|
.action(() => listRedundanciesCLI('remote-videos'))
|
|
|
|
|
|
|
|
program
|
|
|
|
.command('add')
|
|
|
|
.description('Duplicate a video in your redundancy system')
|
|
|
|
.option('-u, --url <url>', 'Server url')
|
|
|
|
.option('-U, --username <username>', 'Username')
|
|
|
|
.option('-p, --password <token>', 'Password')
|
|
|
|
.option('-v, --video <videoId>', 'Video id to duplicate')
|
|
|
|
.action((options) => addRedundancyCLI(options))
|
|
|
|
|
|
|
|
program
|
|
|
|
.command('remove')
|
|
|
|
.description('Remove a video from your redundancies')
|
|
|
|
.option('-u, --url <url>', 'Server url')
|
|
|
|
.option('-U, --username <username>', 'Username')
|
|
|
|
.option('-p, --password <token>', 'Password')
|
|
|
|
.option('-v, --video <videoId>', 'Video id to remove from redundancies')
|
|
|
|
.action((options) => removeRedundancyCLI(options))
|
|
|
|
|
|
|
|
if (!process.argv.slice(2).length) {
|
|
|
|
program.outputHelp()
|
|
|
|
}
|
|
|
|
|
|
|
|
program.parse(process.argv)
|
|
|
|
|
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
|
|
|
|
async function listRedundanciesCLI (target: VideoRedundanciesTarget) {
|
|
|
|
const { url, username, password } = await getServerCredentials(program)
|
|
|
|
const accessToken = await getAdminTokenOrDie(url, username, password)
|
|
|
|
|
|
|
|
const redundancies = await listVideoRedundanciesData(url, accessToken, target)
|
|
|
|
|
|
|
|
const table = new CliTable3({
|
|
|
|
head: [ 'video id', 'video name', 'video url', 'files', 'playlists', 'by instances', 'total size' ]
|
2020-01-31 16:56:52 +01:00
|
|
|
}) as any
|
2020-01-28 11:07:23 +01:00
|
|
|
|
|
|
|
for (const redundancy of redundancies) {
|
|
|
|
const webtorrentFiles = redundancy.redundancies.files
|
|
|
|
const streamingPlaylists = redundancy.redundancies.streamingPlaylists
|
|
|
|
|
|
|
|
let totalSize = ''
|
|
|
|
if (target === 'remote-videos') {
|
|
|
|
const tmp = webtorrentFiles.concat(streamingPlaylists)
|
|
|
|
.reduce((a, b) => a + b.size, 0)
|
|
|
|
|
|
|
|
totalSize = bytes(tmp)
|
|
|
|
}
|
|
|
|
|
|
|
|
const instances = uniq(
|
|
|
|
webtorrentFiles.concat(streamingPlaylists)
|
|
|
|
.map(r => r.fileUrl)
|
2020-01-31 16:56:52 +01:00
|
|
|
.map(u => new URL(u).host)
|
2020-01-28 11:07:23 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
table.push([
|
|
|
|
redundancy.id.toString(),
|
|
|
|
redundancy.name,
|
|
|
|
redundancy.url,
|
|
|
|
webtorrentFiles.length,
|
|
|
|
streamingPlaylists.length,
|
|
|
|
instances.join('\n'),
|
|
|
|
totalSize
|
|
|
|
])
|
|
|
|
}
|
|
|
|
|
|
|
|
console.log(table.toString())
|
|
|
|
process.exit(0)
|
|
|
|
}
|
|
|
|
|
|
|
|
async function addRedundancyCLI (options: { videoId: number }) {
|
|
|
|
const { url, username, password } = await getServerCredentials(program)
|
|
|
|
const accessToken = await getAdminTokenOrDie(url, username, password)
|
|
|
|
|
2020-01-31 16:56:52 +01:00
|
|
|
if (!options['video'] || validator.isInt('' + options['video']) === false) {
|
2020-01-28 11:07:23 +01:00
|
|
|
console.error('You need to specify the video id to duplicate and it should be a number.\n')
|
|
|
|
program.outputHelp()
|
|
|
|
process.exit(-1)
|
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
|
|
|
await addVideoRedundancy({
|
|
|
|
url,
|
|
|
|
accessToken,
|
2020-01-31 16:56:52 +01:00
|
|
|
videoId: options['video']
|
2020-01-28 11:07:23 +01:00
|
|
|
})
|
|
|
|
|
|
|
|
console.log('Video will be duplicated by your instance!')
|
|
|
|
|
|
|
|
process.exit(0)
|
|
|
|
} catch (err) {
|
2020-12-08 21:16:10 +01:00
|
|
|
if (err.message.includes(HttpStatusCode.CONFLICT_409)) {
|
2020-01-28 11:07:23 +01:00
|
|
|
console.error('This video is already duplicated by your instance.')
|
2020-12-08 21:16:10 +01:00
|
|
|
} else if (err.message.includes(HttpStatusCode.NOT_FOUND_404)) {
|
2020-01-28 11:07:23 +01:00
|
|
|
console.error('This video id does not exist.')
|
|
|
|
} else {
|
|
|
|
console.error(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
process.exit(-1)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
async function removeRedundancyCLI (options: { videoId: number }) {
|
|
|
|
const { url, username, password } = await getServerCredentials(program)
|
|
|
|
const accessToken = await getAdminTokenOrDie(url, username, password)
|
|
|
|
|
2020-01-31 16:56:52 +01:00
|
|
|
if (!options['video'] || validator.isInt('' + options['video']) === false) {
|
2020-01-28 11:07:23 +01:00
|
|
|
console.error('You need to specify the video id to remove from your redundancies.\n')
|
|
|
|
program.outputHelp()
|
|
|
|
process.exit(-1)
|
|
|
|
}
|
|
|
|
|
2020-01-31 16:56:52 +01:00
|
|
|
const videoId = parseInt(options['video'] + '', 10)
|
2020-01-28 11:07:23 +01:00
|
|
|
|
|
|
|
let redundancies = await listVideoRedundanciesData(url, accessToken, 'my-videos')
|
|
|
|
let videoRedundancy = redundancies.find(r => videoId === r.id)
|
|
|
|
|
|
|
|
if (!videoRedundancy) {
|
|
|
|
redundancies = await listVideoRedundanciesData(url, accessToken, 'remote-videos')
|
|
|
|
videoRedundancy = redundancies.find(r => videoId === r.id)
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!videoRedundancy) {
|
|
|
|
console.error('Video redundancy not found.')
|
|
|
|
process.exit(-1)
|
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
|
|
|
const ids = videoRedundancy.redundancies.files
|
|
|
|
.concat(videoRedundancy.redundancies.streamingPlaylists)
|
|
|
|
.map(r => r.id)
|
|
|
|
|
|
|
|
for (const id of ids) {
|
|
|
|
await removeVideoRedundancy({
|
|
|
|
url,
|
|
|
|
accessToken,
|
|
|
|
redundancyId: id
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
console.log('Video redundancy removed!')
|
|
|
|
|
|
|
|
process.exit(0)
|
|
|
|
} catch (err) {
|
|
|
|
console.error(err)
|
|
|
|
process.exit(-1)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
async function listVideoRedundanciesData (url: string, accessToken: string, target: VideoRedundanciesTarget) {
|
|
|
|
const res = await listVideoRedundancies({
|
|
|
|
url,
|
|
|
|
accessToken,
|
|
|
|
start: 0,
|
|
|
|
count: 100,
|
|
|
|
sort: 'name',
|
|
|
|
target
|
|
|
|
})
|
|
|
|
|
|
|
|
return res.body.data as VideoRedundancy[]
|
|
|
|
}
|