Plugins can update video constants
Categories, licences and languages
This commit is contained in:
parent
16d5469629
commit
ee286591a5
|
@ -5,7 +5,7 @@ import { CONFIG } from '../../initializers/config'
|
||||||
import { isLibraryCodeValid, isPackageJSONValid } from '../../helpers/custom-validators/plugins'
|
import { isLibraryCodeValid, isPackageJSONValid } from '../../helpers/custom-validators/plugins'
|
||||||
import { ClientScript, PluginPackageJson } from '../../../shared/models/plugins/plugin-package-json.model'
|
import { ClientScript, PluginPackageJson } from '../../../shared/models/plugins/plugin-package-json.model'
|
||||||
import { createReadStream, createWriteStream } from 'fs'
|
import { createReadStream, createWriteStream } from 'fs'
|
||||||
import { PLUGIN_GLOBAL_CSS_PATH } from '../../initializers/constants'
|
import { PLUGIN_GLOBAL_CSS_PATH, VIDEO_CATEGORIES, VIDEO_LANGUAGES, VIDEO_LICENCES } from '../../initializers/constants'
|
||||||
import { PluginType } from '../../../shared/models/plugins/plugin.type'
|
import { PluginType } from '../../../shared/models/plugins/plugin.type'
|
||||||
import { installNpmPlugin, installNpmPluginFromDisk, removeNpmPlugin } from './yarn'
|
import { installNpmPlugin, installNpmPluginFromDisk, removeNpmPlugin } from './yarn'
|
||||||
import { outputFile, readJSON } from 'fs-extra'
|
import { outputFile, readJSON } from 'fs-extra'
|
||||||
|
@ -18,6 +18,9 @@ import { PluginLibrary } from '../../typings/plugins'
|
||||||
import { ClientHtml } from '../client-html'
|
import { ClientHtml } from '../client-html'
|
||||||
import { RegisterServerHookOptions } from '../../../shared/models/plugins/register-server-hook.model'
|
import { RegisterServerHookOptions } from '../../../shared/models/plugins/register-server-hook.model'
|
||||||
import { RegisterServerSettingOptions } from '../../../shared/models/plugins/register-server-setting.model'
|
import { RegisterServerSettingOptions } from '../../../shared/models/plugins/register-server-setting.model'
|
||||||
|
import { PluginVideoLanguageManager } from '../../../shared/models/plugins/plugin-video-language-manager.model'
|
||||||
|
import { PluginVideoCategoryManager } from '../../../shared/models/plugins/plugin-video-category-manager.model'
|
||||||
|
import { PluginVideoLicenceManager } from '../../../shared/models/plugins/plugin-video-licence-manager.model'
|
||||||
|
|
||||||
export interface RegisteredPlugin {
|
export interface RegisteredPlugin {
|
||||||
npmName: string
|
npmName: string
|
||||||
|
@ -46,6 +49,17 @@ export interface HookInformationValue {
|
||||||
priority: number
|
priority: number
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type AlterableVideoConstant = 'language' | 'licence' | 'category'
|
||||||
|
type VideoConstant = { [ key in number | string ]: string }
|
||||||
|
type UpdatedVideoConstant = {
|
||||||
|
[ name in AlterableVideoConstant ]: {
|
||||||
|
[ npmName: string ]: {
|
||||||
|
added: { key: number | string, label: string }[],
|
||||||
|
deleted: { key: number | string, label: string }[]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
export class PluginManager implements ServerHook {
|
export class PluginManager implements ServerHook {
|
||||||
|
|
||||||
private static instance: PluginManager
|
private static instance: PluginManager
|
||||||
|
@ -54,6 +68,12 @@ export class PluginManager implements ServerHook {
|
||||||
private settings: { [ name: string ]: RegisterServerSettingOptions[] } = {}
|
private settings: { [ name: string ]: RegisterServerSettingOptions[] } = {}
|
||||||
private hooks: { [ name: string ]: HookInformationValue[] } = {}
|
private hooks: { [ name: string ]: HookInformationValue[] } = {}
|
||||||
|
|
||||||
|
private updatedVideoConstants: UpdatedVideoConstant = {
|
||||||
|
language: {},
|
||||||
|
licence: {},
|
||||||
|
category: {}
|
||||||
|
}
|
||||||
|
|
||||||
private constructor () {
|
private constructor () {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -161,6 +181,8 @@ export class PluginManager implements ServerHook {
|
||||||
this.hooks[key] = this.hooks[key].filter(h => h.pluginName !== npmName)
|
this.hooks[key] = this.hooks[key].filter(h => h.pluginName !== npmName)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
this.reinitVideoConstants(plugin.npmName)
|
||||||
|
|
||||||
logger.info('Regenerating registered plugin CSS to global file.')
|
logger.info('Regenerating registered plugin CSS to global file.')
|
||||||
await this.regeneratePluginGlobalCSS()
|
await this.regeneratePluginGlobalCSS()
|
||||||
}
|
}
|
||||||
|
@ -427,6 +449,24 @@ export class PluginManager implements ServerHook {
|
||||||
storeData: (key: string, data: any) => PluginModel.storeData(plugin.name, plugin.type, key, data)
|
storeData: (key: string, data: any) => PluginModel.storeData(plugin.name, plugin.type, key, data)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const videoLanguageManager: PluginVideoLanguageManager = {
|
||||||
|
addLanguage: (key: string, label: string) => this.addConstant({ npmName, type: 'language', obj: VIDEO_LANGUAGES, key, label }),
|
||||||
|
|
||||||
|
deleteLanguage: (key: string) => this.deleteConstant({ npmName, type: 'language', obj: VIDEO_LANGUAGES, key })
|
||||||
|
}
|
||||||
|
|
||||||
|
const videoCategoryManager: PluginVideoCategoryManager= {
|
||||||
|
addCategory: (key: number, label: string) => this.addConstant({ npmName, type: 'category', obj: VIDEO_CATEGORIES, key, label }),
|
||||||
|
|
||||||
|
deleteCategory: (key: number) => this.deleteConstant({ npmName, type: 'category', obj: VIDEO_CATEGORIES, key })
|
||||||
|
}
|
||||||
|
|
||||||
|
const videoLicenceManager: PluginVideoLicenceManager = {
|
||||||
|
addLicence: (key: number, label: string) => this.addConstant({ npmName, type: 'licence', obj: VIDEO_LICENCES, key, label }),
|
||||||
|
|
||||||
|
deleteLicence: (key: number) => this.deleteConstant({ npmName, type: 'licence', obj: VIDEO_LICENCES, key })
|
||||||
|
}
|
||||||
|
|
||||||
const peertubeHelpers = {
|
const peertubeHelpers = {
|
||||||
logger
|
logger
|
||||||
}
|
}
|
||||||
|
@ -436,10 +476,90 @@ export class PluginManager implements ServerHook {
|
||||||
registerSetting,
|
registerSetting,
|
||||||
settingsManager,
|
settingsManager,
|
||||||
storageManager,
|
storageManager,
|
||||||
|
videoLanguageManager,
|
||||||
|
videoCategoryManager,
|
||||||
|
videoLicenceManager,
|
||||||
peertubeHelpers
|
peertubeHelpers
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private addConstant <T extends string | number> (parameters: {
|
||||||
|
npmName: string,
|
||||||
|
type: AlterableVideoConstant,
|
||||||
|
obj: VideoConstant,
|
||||||
|
key: T,
|
||||||
|
label: string
|
||||||
|
}) {
|
||||||
|
const { npmName, type, obj, key, label } = parameters
|
||||||
|
|
||||||
|
if (obj[key]) {
|
||||||
|
logger.warn('Cannot add %s %s by plugin %s: key already exists.', type, npmName, key)
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!this.updatedVideoConstants[type][npmName]) {
|
||||||
|
this.updatedVideoConstants[type][npmName] = {
|
||||||
|
added: [],
|
||||||
|
deleted: []
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
this.updatedVideoConstants[type][npmName].added.push({ key, label })
|
||||||
|
obj[key] = label
|
||||||
|
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
private deleteConstant <T extends string | number> (parameters: {
|
||||||
|
npmName: string,
|
||||||
|
type: AlterableVideoConstant,
|
||||||
|
obj: VideoConstant,
|
||||||
|
key: T
|
||||||
|
}) {
|
||||||
|
const { npmName, type, obj, key } = parameters
|
||||||
|
|
||||||
|
if (!obj[key]) {
|
||||||
|
logger.warn('Cannot delete %s %s by plugin %s: key does not exist.', type, npmName, key)
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!this.updatedVideoConstants[type][npmName]) {
|
||||||
|
this.updatedVideoConstants[type][npmName] = {
|
||||||
|
added: [],
|
||||||
|
deleted: []
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
this.updatedVideoConstants[type][npmName].deleted.push({ key, label: obj[key] })
|
||||||
|
delete obj[key]
|
||||||
|
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
private reinitVideoConstants (npmName: string) {
|
||||||
|
const hash = {
|
||||||
|
language: VIDEO_LANGUAGES,
|
||||||
|
licence: VIDEO_LICENCES,
|
||||||
|
category: VIDEO_CATEGORIES
|
||||||
|
}
|
||||||
|
const types: AlterableVideoConstant[] = [ 'language', 'licence', 'category' ]
|
||||||
|
|
||||||
|
for (const type of types) {
|
||||||
|
const updatedConstants = this.updatedVideoConstants[type][npmName]
|
||||||
|
if (!updatedConstants) continue
|
||||||
|
|
||||||
|
for (const added of updatedConstants.added) {
|
||||||
|
delete hash[type][added.key]
|
||||||
|
}
|
||||||
|
|
||||||
|
for (const deleted of updatedConstants.deleted) {
|
||||||
|
hash[type][deleted.key] = deleted.label
|
||||||
|
}
|
||||||
|
|
||||||
|
delete this.updatedVideoConstants[type][npmName]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
static get Instance () {
|
static get Instance () {
|
||||||
return this.instance || (this.instance = new this())
|
return this.instance || (this.instance = new this())
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,39 @@
|
||||||
|
async function register ({
|
||||||
|
registerHook,
|
||||||
|
registerSetting,
|
||||||
|
settingsManager,
|
||||||
|
storageManager,
|
||||||
|
videoCategoryManager,
|
||||||
|
videoLicenceManager,
|
||||||
|
videoLanguageManager
|
||||||
|
}) {
|
||||||
|
videoLanguageManager.addLanguage('al_bhed', 'Al Bhed')
|
||||||
|
videoLanguageManager.addLanguage('al_bhed2', 'Al Bhed 2')
|
||||||
|
videoLanguageManager.deleteLanguage('en')
|
||||||
|
videoLanguageManager.deleteLanguage('fr')
|
||||||
|
|
||||||
|
videoCategoryManager.addCategory(42, 'Best category')
|
||||||
|
videoCategoryManager.addCategory(43, 'High best category')
|
||||||
|
videoCategoryManager.deleteCategory(1) // Music
|
||||||
|
videoCategoryManager.deleteCategory(2) // Films
|
||||||
|
|
||||||
|
videoLicenceManager.addLicence(42, 'Best licence')
|
||||||
|
videoLicenceManager.addLicence(43, 'High best licence')
|
||||||
|
videoLicenceManager.deleteLicence(1) // Attribution
|
||||||
|
videoLicenceManager.deleteLicence(7) // Public domain
|
||||||
|
}
|
||||||
|
|
||||||
|
async function unregister () {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
register,
|
||||||
|
unregister
|
||||||
|
}
|
||||||
|
|
||||||
|
// ############################################################################
|
||||||
|
|
||||||
|
function addToCount (obj) {
|
||||||
|
return Object.assign({}, obj, { count: obj.count + 1 })
|
||||||
|
}
|
|
@ -0,0 +1,19 @@
|
||||||
|
{
|
||||||
|
"name": "peertube-plugin-test-three",
|
||||||
|
"version": "0.0.1",
|
||||||
|
"description": "Plugin test 3",
|
||||||
|
"engine": {
|
||||||
|
"peertube": ">=1.3.0"
|
||||||
|
},
|
||||||
|
"keywords": [
|
||||||
|
"peertube",
|
||||||
|
"plugin"
|
||||||
|
],
|
||||||
|
"homepage": "https://github.com/Chocobozzz/PeerTube",
|
||||||
|
"author": "Chocobozzz",
|
||||||
|
"bugs": "https://github.com/Chocobozzz/PeerTube/issues",
|
||||||
|
"library": "./main.js",
|
||||||
|
"staticDirs": {},
|
||||||
|
"css": [],
|
||||||
|
"clientScripts": []
|
||||||
|
}
|
|
@ -1,2 +1,3 @@
|
||||||
import './action-hooks'
|
import './action-hooks'
|
||||||
import './filter-hooks'
|
import './filter-hooks'
|
||||||
|
import './video-constants'
|
||||||
|
|
|
@ -0,0 +1,140 @@
|
||||||
|
/* tslint:disable:no-unused-expression */
|
||||||
|
|
||||||
|
import * as chai from 'chai'
|
||||||
|
import 'mocha'
|
||||||
|
import {
|
||||||
|
cleanupTests,
|
||||||
|
flushAndRunMultipleServers,
|
||||||
|
flushAndRunServer, killallServers, reRunServer,
|
||||||
|
ServerInfo,
|
||||||
|
waitUntilLog
|
||||||
|
} from '../../../shared/extra-utils/server/servers'
|
||||||
|
import {
|
||||||
|
addVideoCommentReply,
|
||||||
|
addVideoCommentThread,
|
||||||
|
deleteVideoComment,
|
||||||
|
getPluginTestPath,
|
||||||
|
getVideosList,
|
||||||
|
installPlugin,
|
||||||
|
removeVideo,
|
||||||
|
setAccessTokensToServers,
|
||||||
|
updateVideo,
|
||||||
|
uploadVideo,
|
||||||
|
viewVideo,
|
||||||
|
getVideosListPagination,
|
||||||
|
getVideo,
|
||||||
|
getVideoCommentThreads,
|
||||||
|
getVideoThreadComments,
|
||||||
|
getVideoWithToken,
|
||||||
|
setDefaultVideoChannel,
|
||||||
|
waitJobs,
|
||||||
|
doubleFollow, getVideoLanguages, getVideoLicences, getVideoCategories, uninstallPlugin
|
||||||
|
} from '../../../shared/extra-utils'
|
||||||
|
import { VideoCommentThreadTree } from '../../../shared/models/videos/video-comment.model'
|
||||||
|
import { VideoDetails } from '../../../shared/models/videos'
|
||||||
|
import { getYoutubeVideoUrl, importVideo } from '../../../shared/extra-utils/videos/video-imports'
|
||||||
|
|
||||||
|
const expect = chai.expect
|
||||||
|
|
||||||
|
describe('Test plugin altering video constants', function () {
|
||||||
|
let server: ServerInfo
|
||||||
|
|
||||||
|
before(async function () {
|
||||||
|
this.timeout(30000)
|
||||||
|
|
||||||
|
server = await flushAndRunServer(1)
|
||||||
|
await setAccessTokensToServers([ server ])
|
||||||
|
|
||||||
|
await installPlugin({
|
||||||
|
url: server.url,
|
||||||
|
accessToken: server.accessToken,
|
||||||
|
path: getPluginTestPath('-three')
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should have updated languages', async function () {
|
||||||
|
const res = await getVideoLanguages(server.url)
|
||||||
|
const languages = res.body
|
||||||
|
|
||||||
|
expect(languages['en']).to.not.exist
|
||||||
|
expect(languages['fr']).to.not.exist
|
||||||
|
|
||||||
|
expect(languages['al_bhed']).to.equal('Al Bhed')
|
||||||
|
expect(languages['al_bhed2']).to.equal('Al Bhed 2')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should have updated categories', async function () {
|
||||||
|
const res = await getVideoCategories(server.url)
|
||||||
|
const categories = res.body
|
||||||
|
|
||||||
|
expect(categories[1]).to.not.exist
|
||||||
|
expect(categories[2]).to.not.exist
|
||||||
|
|
||||||
|
expect(categories[42]).to.equal('Best category')
|
||||||
|
expect(categories[43]).to.equal('High best category')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should have updated licences', async function () {
|
||||||
|
const res = await getVideoLicences(server.url)
|
||||||
|
const licences = res.body
|
||||||
|
|
||||||
|
expect(licences[1]).to.not.exist
|
||||||
|
expect(licences[7]).to.not.exist
|
||||||
|
|
||||||
|
expect(licences[42]).to.equal('Best licence')
|
||||||
|
expect(licences[43]).to.equal('High best licence')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should be able to upload a video with these values', async function () {
|
||||||
|
const attrs = { name: 'video', category: 42, licence: 42, language: 'al_bhed2' }
|
||||||
|
const resUpload = await uploadVideo(server.url, server.accessToken, attrs)
|
||||||
|
|
||||||
|
const res = await getVideo(server.url, resUpload.body.video.uuid)
|
||||||
|
|
||||||
|
const video: VideoDetails = res.body
|
||||||
|
expect(video.language.label).to.equal('Al Bhed 2')
|
||||||
|
expect(video.licence.label).to.equal('Best licence')
|
||||||
|
expect(video.category.label).to.equal('Best category')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should uninstall the plugin and reset languages, categories and licences', async function () {
|
||||||
|
await uninstallPlugin({ url: server.url, accessToken: server.accessToken, npmName: 'peertube-plugin-test-three' })
|
||||||
|
|
||||||
|
{
|
||||||
|
const res = await getVideoLanguages(server.url)
|
||||||
|
const languages = res.body
|
||||||
|
|
||||||
|
expect(languages[ 'en' ]).to.equal('English')
|
||||||
|
expect(languages[ 'fr' ]).to.equal('French')
|
||||||
|
|
||||||
|
expect(languages[ 'al_bhed' ]).to.not.exist
|
||||||
|
expect(languages[ 'al_bhed2' ]).to.not.exist
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
const res = await getVideoCategories(server.url)
|
||||||
|
const categories = res.body
|
||||||
|
|
||||||
|
expect(categories[ 1 ]).to.equal('Music')
|
||||||
|
expect(categories[ 2 ]).to.equal('Films')
|
||||||
|
|
||||||
|
expect(categories[ 42 ]).to.not.exist
|
||||||
|
expect(categories[ 43 ]).to.not.exist
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
const res = await getVideoLicences(server.url)
|
||||||
|
const licences = res.body
|
||||||
|
|
||||||
|
expect(licences[ 1 ]).to.equal('Attribution')
|
||||||
|
expect(licences[ 7 ]).to.equal('Public Domain Dedication')
|
||||||
|
|
||||||
|
expect(licences[ 42 ]).to.not.exist
|
||||||
|
expect(licences[ 43 ]).to.not.exist
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
after(async function () {
|
||||||
|
await cleanupTests([ server ])
|
||||||
|
})
|
||||||
|
})
|
|
@ -3,6 +3,9 @@ import { PluginSettingsManager } from '../../../shared/models/plugins/plugin-set
|
||||||
import { PluginStorageManager } from '../../../shared/models/plugins/plugin-storage-manager.model'
|
import { PluginStorageManager } from '../../../shared/models/plugins/plugin-storage-manager.model'
|
||||||
import { RegisterServerHookOptions } from '../../../shared/models/plugins/register-server-hook.model'
|
import { RegisterServerHookOptions } from '../../../shared/models/plugins/register-server-hook.model'
|
||||||
import { RegisterServerSettingOptions } from '../../../shared/models/plugins/register-server-setting.model'
|
import { RegisterServerSettingOptions } from '../../../shared/models/plugins/register-server-setting.model'
|
||||||
|
import { PluginVideoCategoryManager } from '../../../shared/models/plugins/plugin-video-category-manager.model'
|
||||||
|
import { PluginVideoLanguageManager } from '../../../shared/models/plugins/plugin-video-language-manager.model'
|
||||||
|
import { PluginVideoLicenceManager } from '../../../shared/models/plugins/plugin-video-licence-manager.model'
|
||||||
|
|
||||||
export type RegisterServerOptions = {
|
export type RegisterServerOptions = {
|
||||||
registerHook: (options: RegisterServerHookOptions) => void
|
registerHook: (options: RegisterServerHookOptions) => void
|
||||||
|
@ -13,6 +16,10 @@ export type RegisterServerOptions = {
|
||||||
|
|
||||||
storageManager: PluginStorageManager
|
storageManager: PluginStorageManager
|
||||||
|
|
||||||
|
videoCategoryManager: PluginVideoCategoryManager
|
||||||
|
videoLanguageManager: PluginVideoLanguageManager
|
||||||
|
videoLicenceManager: PluginVideoLicenceManager
|
||||||
|
|
||||||
peertubeHelpers: {
|
peertubeHelpers: {
|
||||||
logger: typeof logger
|
logger: typeof logger
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,5 @@
|
||||||
|
export interface PluginVideoCategoryManager {
|
||||||
|
addCategory: (categoryKey: number, categoryLabel: string) => boolean
|
||||||
|
|
||||||
|
deleteCategory: (categoryKey: number) => boolean
|
||||||
|
}
|
|
@ -0,0 +1,5 @@
|
||||||
|
export interface PluginVideoLanguageManager {
|
||||||
|
addLanguage: (languageKey: string, languageLabel: string) => boolean
|
||||||
|
|
||||||
|
deleteLanguage: (languageKey: string) => boolean
|
||||||
|
}
|
|
@ -0,0 +1,5 @@
|
||||||
|
export interface PluginVideoLicenceManager {
|
||||||
|
addLicence: (licenceKey: number, licenceLabel: string) => boolean
|
||||||
|
|
||||||
|
deleteLicence: (licenceKey: number) => boolean
|
||||||
|
}
|
Loading…
Reference in New Issue