Add plugin API tests
This commit is contained in:
parent
9b474844e8
commit
09071200c7
|
@ -21,6 +21,7 @@ storage:
|
||||||
torrents: 'test1/torrents/'
|
torrents: 'test1/torrents/'
|
||||||
captions: 'test1/captions/'
|
captions: 'test1/captions/'
|
||||||
cache: 'test1/cache/'
|
cache: 'test1/cache/'
|
||||||
|
plugins: 'test1/plugins/'
|
||||||
|
|
||||||
admin:
|
admin:
|
||||||
email: 'admin1@example.com'
|
email: 'admin1@example.com'
|
||||||
|
|
|
@ -21,6 +21,7 @@ storage:
|
||||||
torrents: 'test2/torrents/'
|
torrents: 'test2/torrents/'
|
||||||
captions: 'test2/captions/'
|
captions: 'test2/captions/'
|
||||||
cache: 'test2/cache/'
|
cache: 'test2/cache/'
|
||||||
|
plugins: 'test2/plugins/'
|
||||||
|
|
||||||
admin:
|
admin:
|
||||||
email: 'admin2@example.com'
|
email: 'admin2@example.com'
|
||||||
|
|
|
@ -21,6 +21,7 @@ storage:
|
||||||
torrents: 'test3/torrents/'
|
torrents: 'test3/torrents/'
|
||||||
captions: 'test3/captions/'
|
captions: 'test3/captions/'
|
||||||
cache: 'test3/cache/'
|
cache: 'test3/cache/'
|
||||||
|
plugins: 'test3/plugins/'
|
||||||
|
|
||||||
admin:
|
admin:
|
||||||
email: 'admin3@example.com'
|
email: 'admin3@example.com'
|
||||||
|
|
|
@ -21,6 +21,7 @@ storage:
|
||||||
torrents: 'test4/torrents/'
|
torrents: 'test4/torrents/'
|
||||||
captions: 'test4/captions/'
|
captions: 'test4/captions/'
|
||||||
cache: 'test4/cache/'
|
cache: 'test4/cache/'
|
||||||
|
plugins: 'test4/plugins/'
|
||||||
|
|
||||||
admin:
|
admin:
|
||||||
email: 'admin4@example.com'
|
email: 'admin4@example.com'
|
||||||
|
|
|
@ -21,6 +21,7 @@ storage:
|
||||||
torrents: 'test5/torrents/'
|
torrents: 'test5/torrents/'
|
||||||
captions: 'test5/captions/'
|
captions: 'test5/captions/'
|
||||||
cache: 'test5/cache/'
|
cache: 'test5/cache/'
|
||||||
|
plugins: 'test5/plugins/'
|
||||||
|
|
||||||
admin:
|
admin:
|
||||||
email: 'admin5@example.com'
|
email: 'admin5@example.com'
|
||||||
|
|
|
@ -21,6 +21,7 @@ storage:
|
||||||
torrents: 'test6/torrents/'
|
torrents: 'test6/torrents/'
|
||||||
captions: 'test6/captions/'
|
captions: 'test6/captions/'
|
||||||
cache: 'test6/cache/'
|
cache: 'test6/cache/'
|
||||||
|
plugins: 'test6/plugins/'
|
||||||
|
|
||||||
admin:
|
admin:
|
||||||
email: 'admin6@example.com'
|
email: 'admin6@example.com'
|
||||||
|
|
|
@ -25,6 +25,7 @@ import { ManagePlugin } from '../../../shared/models/plugins/manage-plugin.model
|
||||||
import { logger } from '../../helpers/logger'
|
import { logger } from '../../helpers/logger'
|
||||||
import { listAvailablePluginsFromIndex } from '../../lib/plugins/plugin-index'
|
import { listAvailablePluginsFromIndex } from '../../lib/plugins/plugin-index'
|
||||||
import { PeertubePluginIndexList } from '../../../shared/models/plugins/peertube-plugin-index-list.model'
|
import { PeertubePluginIndexList } from '../../../shared/models/plugins/peertube-plugin-index-list.model'
|
||||||
|
import { RegisteredSettings } from '../../../shared/models/plugins/register-setting.model'
|
||||||
|
|
||||||
const pluginRouter = express.Router()
|
const pluginRouter = express.Router()
|
||||||
|
|
||||||
|
@ -103,9 +104,11 @@ export {
|
||||||
|
|
||||||
async function listPlugins (req: express.Request, res: express.Response) {
|
async function listPlugins (req: express.Request, res: express.Response) {
|
||||||
const pluginType = req.query.pluginType
|
const pluginType = req.query.pluginType
|
||||||
|
const uninstalled = req.query.uninstalled
|
||||||
|
|
||||||
const resultList = await PluginModel.listForApi({
|
const resultList = await PluginModel.listForApi({
|
||||||
pluginType,
|
pluginType,
|
||||||
|
uninstalled,
|
||||||
start: req.query.start,
|
start: req.query.start,
|
||||||
count: req.query.count,
|
count: req.query.count,
|
||||||
sort: req.query.sort
|
sort: req.query.sort
|
||||||
|
@ -161,9 +164,9 @@ async function uninstallPlugin (req: express.Request, res: express.Response) {
|
||||||
function getPluginRegisteredSettings (req: express.Request, res: express.Response) {
|
function getPluginRegisteredSettings (req: express.Request, res: express.Response) {
|
||||||
const settings = PluginManager.Instance.getRegisteredSettings(req.params.npmName)
|
const settings = PluginManager.Instance.getRegisteredSettings(req.params.npmName)
|
||||||
|
|
||||||
return res.json({
|
const json: RegisteredSettings = { settings }
|
||||||
settings
|
|
||||||
})
|
return res.json(json)
|
||||||
}
|
}
|
||||||
|
|
||||||
async function updatePluginSettings (req: express.Request, res: express.Response) {
|
async function updatePluginSettings (req: express.Request, res: express.Response) {
|
||||||
|
|
|
@ -21,7 +21,7 @@ async function listAvailablePluginsFromIndex (options: PeertubePluginIndexList)
|
||||||
sort,
|
sort,
|
||||||
pluginType,
|
pluginType,
|
||||||
search,
|
search,
|
||||||
currentPeerTubeEngine: PEERTUBE_VERSION
|
currentPeerTubeEngine: options.currentPeerTubeEngine || PEERTUBE_VERSION
|
||||||
}
|
}
|
||||||
|
|
||||||
const uri = CONFIG.PLUGINS.INDEX.URL + '/api/v1/plugins'
|
const uri = CONFIG.PLUGINS.INDEX.URL + '/api/v1/plugins'
|
||||||
|
|
|
@ -8,7 +8,7 @@ import { createReadStream, createWriteStream } from 'fs'
|
||||||
import { PLUGIN_GLOBAL_CSS_PATH } from '../../initializers/constants'
|
import { PLUGIN_GLOBAL_CSS_PATH } 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 } from 'fs-extra'
|
import { outputFile, readJSON } from 'fs-extra'
|
||||||
import { RegisterSettingOptions } from '../../../shared/models/plugins/register-setting.model'
|
import { RegisterSettingOptions } from '../../../shared/models/plugins/register-setting.model'
|
||||||
import { RegisterHookOptions } from '../../../shared/models/plugins/register-hook.model'
|
import { RegisterHookOptions } from '../../../shared/models/plugins/register-hook.model'
|
||||||
import { PluginSettingsManager } from '../../../shared/models/plugins/plugin-settings-manager.model'
|
import { PluginSettingsManager } from '../../../shared/models/plugins/plugin-settings-manager.model'
|
||||||
|
@ -174,7 +174,7 @@ export class PluginManager implements ServerHook {
|
||||||
const pluginType = PluginModel.getTypeFromNpmName(npmName)
|
const pluginType = PluginModel.getTypeFromNpmName(npmName)
|
||||||
const pluginName = PluginModel.normalizePluginName(npmName)
|
const pluginName = PluginModel.normalizePluginName(npmName)
|
||||||
|
|
||||||
const packageJSON = this.getPackageJSON(pluginName, pluginType)
|
const packageJSON = await this.getPackageJSON(pluginName, pluginType)
|
||||||
if (!isPackageJSONValid(packageJSON, pluginType)) {
|
if (!isPackageJSONValid(packageJSON, pluginType)) {
|
||||||
throw new Error('PackageJSON is invalid.')
|
throw new Error('PackageJSON is invalid.')
|
||||||
}
|
}
|
||||||
|
@ -251,7 +251,7 @@ export class PluginManager implements ServerHook {
|
||||||
|
|
||||||
logger.info('Registering plugin or theme %s.', npmName)
|
logger.info('Registering plugin or theme %s.', npmName)
|
||||||
|
|
||||||
const packageJSON = this.getPackageJSON(plugin.name, plugin.type)
|
const packageJSON = await this.getPackageJSON(plugin.name, plugin.type)
|
||||||
const pluginPath = this.getPluginPath(plugin.name, plugin.type)
|
const pluginPath = this.getPluginPath(plugin.name, plugin.type)
|
||||||
|
|
||||||
if (!isPackageJSONValid(packageJSON, plugin.type)) {
|
if (!isPackageJSONValid(packageJSON, plugin.type)) {
|
||||||
|
@ -286,7 +286,10 @@ export class PluginManager implements ServerHook {
|
||||||
private async registerPlugin (plugin: PluginModel, pluginPath: string, packageJSON: PluginPackageJson) {
|
private async registerPlugin (plugin: PluginModel, pluginPath: string, packageJSON: PluginPackageJson) {
|
||||||
const npmName = PluginModel.buildNpmName(plugin.name, plugin.type)
|
const npmName = PluginModel.buildNpmName(plugin.name, plugin.type)
|
||||||
|
|
||||||
const library: PluginLibrary = require(join(pluginPath, packageJSON.library))
|
// Delete cache if needed
|
||||||
|
const modulePath = join(pluginPath, packageJSON.library)
|
||||||
|
delete require.cache[modulePath]
|
||||||
|
const library: PluginLibrary = require(modulePath)
|
||||||
|
|
||||||
if (!isLibraryCodeValid(library)) {
|
if (!isLibraryCodeValid(library)) {
|
||||||
throw new Error('Library code is not valid (miss register or unregister function)')
|
throw new Error('Library code is not valid (miss register or unregister function)')
|
||||||
|
@ -350,7 +353,7 @@ export class PluginManager implements ServerHook {
|
||||||
private getPackageJSON (pluginName: string, pluginType: PluginType) {
|
private getPackageJSON (pluginName: string, pluginType: PluginType) {
|
||||||
const pluginPath = join(this.getPluginPath(pluginName, pluginType), 'package.json')
|
const pluginPath = join(this.getPluginPath(pluginName, pluginType), 'package.json')
|
||||||
|
|
||||||
return require(pluginPath) as PluginPackageJson
|
return readJSON(pluginPath) as Promise<PluginPackageJson>
|
||||||
}
|
}
|
||||||
|
|
||||||
private getPluginPath (pluginName: string, pluginType: PluginType) {
|
private getPluginPath (pluginName: string, pluginType: PluginType) {
|
||||||
|
|
|
@ -13,7 +13,9 @@ async function installNpmPlugin (npmName: string, version?: string) {
|
||||||
let toInstall = npmName
|
let toInstall = npmName
|
||||||
if (version) toInstall += `@${version}`
|
if (version) toInstall += `@${version}`
|
||||||
|
|
||||||
await execYarn('add ' + toInstall)
|
const { stdout } = await execYarn('add ' + toInstall)
|
||||||
|
|
||||||
|
logger.debug('Added a yarn package.', { yarnStdout: stdout })
|
||||||
}
|
}
|
||||||
|
|
||||||
async function installNpmPluginFromDisk (path: string) {
|
async function installNpmPluginFromDisk (path: string) {
|
||||||
|
@ -46,7 +48,7 @@ async function execYarn (command: string) {
|
||||||
await outputJSON(pluginPackageJSON, {})
|
await outputJSON(pluginPackageJSON, {})
|
||||||
}
|
}
|
||||||
|
|
||||||
await execShell(`yarn ${command}`, { cwd: pluginDirectory })
|
return execShell(`yarn ${command}`, { cwd: pluginDirectory })
|
||||||
} catch (result) {
|
} catch (result) {
|
||||||
logger.error('Cannot exec yarn.', { command, err: result.err, stderr: result.stderr })
|
logger.error('Cannot exec yarn.', { command, err: result.err, stderr: result.stderr })
|
||||||
|
|
||||||
|
|
|
@ -127,6 +127,9 @@ const listAvailablePluginsValidator = [
|
||||||
query('pluginType')
|
query('pluginType')
|
||||||
.optional()
|
.optional()
|
||||||
.custom(isPluginTypeValid).withMessage('Should have a valid plugin type'),
|
.custom(isPluginTypeValid).withMessage('Should have a valid plugin type'),
|
||||||
|
query('currentPeerTubeEngine')
|
||||||
|
.optional()
|
||||||
|
.custom(isPluginVersionValid).withMessage('Should have a valid current peertube engine'),
|
||||||
|
|
||||||
(req: express.Request, res: express.Response, next: express.NextFunction) => {
|
(req: express.Request, res: express.Response, next: express.NextFunction) => {
|
||||||
logger.debug('Checking enabledPluginValidator parameters', { parameters: req.query })
|
logger.debug('Checking enabledPluginValidator parameters', { parameters: req.query })
|
||||||
|
|
|
@ -152,7 +152,8 @@ describe('Test server plugins API validators', function () {
|
||||||
const path = '/api/v1/plugins/available'
|
const path = '/api/v1/plugins/available'
|
||||||
const baseQuery = {
|
const baseQuery = {
|
||||||
search: 'super search',
|
search: 'super search',
|
||||||
pluginType: PluginType.PLUGIN
|
pluginType: PluginType.PLUGIN,
|
||||||
|
currentPeerTubeEngine: '1.2.3'
|
||||||
}
|
}
|
||||||
|
|
||||||
it('Should fail with an invalid token', async function () {
|
it('Should fail with an invalid token', async function () {
|
||||||
|
@ -198,6 +199,17 @@ describe('Test server plugins API validators', function () {
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
|
it('Should fail with an invalid current peertube engine', async function () {
|
||||||
|
const query = immutableAssign(baseQuery, { currentPeerTubeEngine: '1.0' })
|
||||||
|
|
||||||
|
await makeGetRequest({
|
||||||
|
url: server.url,
|
||||||
|
path,
|
||||||
|
token: server.accessToken,
|
||||||
|
query
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
it('Should success with the correct parameters', async function () {
|
it('Should success with the correct parameters', async function () {
|
||||||
await makeGetRequest({
|
await makeGetRequest({
|
||||||
url: server.url,
|
url: server.url,
|
||||||
|
|
|
@ -2,129 +2,416 @@
|
||||||
|
|
||||||
import 'mocha'
|
import 'mocha'
|
||||||
import * as chai from 'chai'
|
import * as chai from 'chai'
|
||||||
import { About } from '../../../../shared/models/server/about.model'
|
|
||||||
import { CustomConfig } from '../../../../shared/models/server/custom-config.model'
|
|
||||||
import {
|
import {
|
||||||
cleanupTests,
|
cleanupTests,
|
||||||
deleteCustomConfig,
|
closeAllSequelize,
|
||||||
flushAndRunServer,
|
flushAndRunServer,
|
||||||
getAbout,
|
getConfig, getMyUserInformation, getPluginPackageJSON,
|
||||||
getConfig,
|
getPlugin,
|
||||||
getCustomConfig, installPlugin,
|
getPluginRegisteredSettings,
|
||||||
killallServers, parallelTests,
|
getPluginsCSS,
|
||||||
registerUser,
|
installPlugin, killallServers,
|
||||||
reRunServer, ServerInfo,
|
listAvailablePlugins,
|
||||||
|
listPlugins, reRunServer,
|
||||||
|
ServerInfo,
|
||||||
setAccessTokensToServers,
|
setAccessTokensToServers,
|
||||||
updateCustomConfig, uploadVideo
|
setPluginVersion, uninstallPlugin,
|
||||||
|
updateCustomSubConfig, updateMyUser, updatePluginPackageJSON, updatePlugin,
|
||||||
|
updatePluginSettings,
|
||||||
|
wait
|
||||||
} from '../../../../shared/extra-utils'
|
} from '../../../../shared/extra-utils'
|
||||||
import { ServerConfig } from '../../../../shared/models'
|
import { PluginType } from '../../../../shared/models/plugins/plugin.type'
|
||||||
|
import { PeerTubePluginIndex } from '../../../../shared/models/plugins/peertube-plugin-index.model'
|
||||||
|
import { ServerConfig } from '../../../../shared/models/server'
|
||||||
|
import { RegisteredSettings } from '../../../../shared/models/plugins/register-setting.model'
|
||||||
import { PeerTubePlugin } from '../../../../shared/models/plugins/peertube-plugin.model'
|
import { PeerTubePlugin } from '../../../../shared/models/plugins/peertube-plugin.model'
|
||||||
|
import { User } from '../../../../shared/models/users'
|
||||||
|
import { PluginPackageJson } from '../../../../shared/models/plugins/plugin-package-json.model'
|
||||||
|
|
||||||
const expect = chai.expect
|
const expect = chai.expect
|
||||||
|
|
||||||
describe('Test plugins', function () {
|
describe('Test plugins', function () {
|
||||||
let server = null
|
let server: ServerInfo = null
|
||||||
|
|
||||||
before(async function () {
|
before(async function () {
|
||||||
this.timeout(30000)
|
this.timeout(30000)
|
||||||
|
|
||||||
server = await flushAndRunServer(1)
|
const configOverride = {
|
||||||
|
plugins: {
|
||||||
|
index: { check_latest_versions_interval: '5 seconds' }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
server = await flushAndRunServer(1, configOverride)
|
||||||
await setAccessTokensToServers([ server ])
|
await setAccessTokensToServers([ server ])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should list and search available plugins and themes', async function () {
|
||||||
|
this.timeout(30000)
|
||||||
|
|
||||||
{
|
{
|
||||||
await installPlugin({ url: server.url, accessToken: server.accessToken, npmName: 'peertube-plugin-hello-world' })
|
const res = await listAvailablePlugins({
|
||||||
|
url: server.url,
|
||||||
|
accessToken: server.accessToken,
|
||||||
|
count: 1,
|
||||||
|
start: 0,
|
||||||
|
pluginType: PluginType.THEME,
|
||||||
|
search: 'background-red'
|
||||||
|
})
|
||||||
|
|
||||||
|
expect(res.body.total).to.be.at.least(1)
|
||||||
|
expect(res.body.data).to.have.lengthOf(1)
|
||||||
}
|
}
|
||||||
|
|
||||||
{
|
{
|
||||||
await installPlugin({ url: server.url, accessToken: server.accessToken, npmName: 'peertube-plugin-background-color' })
|
const res1 = await listAvailablePlugins({
|
||||||
|
url: server.url,
|
||||||
|
accessToken: server.accessToken,
|
||||||
|
count: 2,
|
||||||
|
start: 0,
|
||||||
|
sort: 'npmName'
|
||||||
|
})
|
||||||
|
const data1: PeerTubePluginIndex[] = res1.body.data
|
||||||
|
|
||||||
|
expect(res1.body.total).to.be.at.least(2)
|
||||||
|
expect(data1).to.have.lengthOf(2)
|
||||||
|
|
||||||
|
const res2 = await listAvailablePlugins({
|
||||||
|
url: server.url,
|
||||||
|
accessToken: server.accessToken,
|
||||||
|
count: 2,
|
||||||
|
start: 0,
|
||||||
|
sort: '-npmName'
|
||||||
|
})
|
||||||
|
const data2: PeerTubePluginIndex[] = res2.body.data
|
||||||
|
|
||||||
|
expect(res2.body.total).to.be.at.least(2)
|
||||||
|
expect(data2).to.have.lengthOf(2)
|
||||||
|
|
||||||
|
expect(data1[0].npmName).to.not.equal(data2[ 0 ].npmName)
|
||||||
}
|
}
|
||||||
})
|
|
||||||
|
|
||||||
it('Should list available plugins and themes', async function () {
|
{
|
||||||
// List without filter
|
const res = await listAvailablePlugins({
|
||||||
// List with filter (plugin and theme)
|
url: server.url,
|
||||||
})
|
accessToken: server.accessToken,
|
||||||
|
count: 10,
|
||||||
|
start: 0,
|
||||||
|
pluginType: PluginType.THEME,
|
||||||
|
search: 'background-red',
|
||||||
|
currentPeerTubeEngine: '1.0.0'
|
||||||
|
})
|
||||||
|
const data: PeerTubePluginIndex[] = res.body.data
|
||||||
|
|
||||||
it('Should search available plugins', async function () {
|
const p = data.find(p => p.npmName === 'peertube-theme-background-red')
|
||||||
// Search with filter (plugin and theme)
|
expect(p).to.be.undefined
|
||||||
// Add pagination
|
}
|
||||||
// Add sort
|
|
||||||
// Add peertube engine
|
|
||||||
})
|
})
|
||||||
|
|
||||||
it('Should have an empty global css', async function () {
|
it('Should have an empty global css', async function () {
|
||||||
// get /global.css
|
const res = await getPluginsCSS(server.url)
|
||||||
|
|
||||||
|
expect(res.text).to.be.empty
|
||||||
})
|
})
|
||||||
|
|
||||||
it('Should install a plugin and a theme', async function () {
|
it('Should install a plugin and a theme', async function () {
|
||||||
|
this.timeout(30000)
|
||||||
|
|
||||||
|
await installPlugin({
|
||||||
|
url: server.url,
|
||||||
|
accessToken: server.accessToken,
|
||||||
|
npmName: 'peertube-plugin-hello-world'
|
||||||
|
})
|
||||||
|
|
||||||
|
await installPlugin({
|
||||||
|
url: server.url,
|
||||||
|
accessToken: server.accessToken,
|
||||||
|
npmName: 'peertube-theme-background-red'
|
||||||
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
it('Should have the correct global css', async function () {
|
it('Should have the correct global css', async function () {
|
||||||
// get /global.css
|
const res = await getPluginsCSS(server.url)
|
||||||
|
|
||||||
|
expect(res.text).to.contain('--mainBackgroundColor')
|
||||||
})
|
})
|
||||||
|
|
||||||
it('Should have the plugin loaded in the configuration', async function () {
|
it('Should have the plugin loaded in the configuration', async function () {
|
||||||
// Check registered themes/plugins
|
const res = await getConfig(server.url)
|
||||||
|
const config: ServerConfig = res.body
|
||||||
|
|
||||||
|
const theme = config.theme.registered.find(r => r.name === 'background-red')
|
||||||
|
expect(theme).to.not.be.undefined
|
||||||
|
|
||||||
|
const plugin = config.plugin.registered.find(r => r.name === 'hello-world')
|
||||||
|
expect(plugin).to.not.be.undefined
|
||||||
})
|
})
|
||||||
|
|
||||||
it('Should update the default theme in the configuration', async function () {
|
it('Should update the default theme in the configuration', async function () {
|
||||||
// Update config
|
await updateCustomSubConfig(server.url, server.accessToken, { theme: { default: 'background-red' } })
|
||||||
|
|
||||||
|
const res = await getConfig(server.url)
|
||||||
|
const config: ServerConfig = res.body
|
||||||
|
|
||||||
|
expect(config.theme.default).to.equal('background-red')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should update my default theme', async function () {
|
||||||
|
await updateMyUser({
|
||||||
|
url: server.url,
|
||||||
|
accessToken: server.accessToken,
|
||||||
|
theme: 'background-red'
|
||||||
|
})
|
||||||
|
|
||||||
|
const res = await getMyUserInformation(server.url, server.accessToken)
|
||||||
|
expect((res.body as User).theme).to.equal('background-red')
|
||||||
})
|
})
|
||||||
|
|
||||||
it('Should list plugins and themes', async function () {
|
it('Should list plugins and themes', async function () {
|
||||||
// List without filter
|
{
|
||||||
// List with filter (theme/plugin)
|
const res = await listPlugins({
|
||||||
// List with pagination
|
url: server.url,
|
||||||
// List with sort
|
accessToken: server.accessToken,
|
||||||
})
|
count: 1,
|
||||||
|
start: 0,
|
||||||
|
pluginType: PluginType.THEME
|
||||||
|
})
|
||||||
|
const data: PeerTubePlugin[] = res.body.data
|
||||||
|
|
||||||
it('Should get a plugin and a theme', async function () {
|
expect(res.body.total).to.be.at.least(1)
|
||||||
// Get plugin
|
expect(data).to.have.lengthOf(1)
|
||||||
// Get theme
|
expect(data[0].name).to.equal('background-red')
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
const res = await listPlugins({
|
||||||
|
url: server.url,
|
||||||
|
accessToken: server.accessToken,
|
||||||
|
count: 2,
|
||||||
|
start: 0,
|
||||||
|
sort: 'name'
|
||||||
|
})
|
||||||
|
const data: PeerTubePlugin[] = res.body.data
|
||||||
|
|
||||||
|
expect(data[0].name).to.equal('background-red')
|
||||||
|
expect(data[1].name).to.equal('hello-world')
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
const res = await listPlugins({
|
||||||
|
url: server.url,
|
||||||
|
accessToken: server.accessToken,
|
||||||
|
count: 2,
|
||||||
|
start: 1,
|
||||||
|
sort: 'name'
|
||||||
|
})
|
||||||
|
const data: PeerTubePlugin[] = res.body.data
|
||||||
|
|
||||||
|
expect(data[0].name).to.equal('hello-world')
|
||||||
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
it('Should get registered settings', async function () {
|
it('Should get registered settings', async function () {
|
||||||
// Get plugin
|
const res = await getPluginRegisteredSettings({
|
||||||
|
url: server.url,
|
||||||
|
accessToken: server.accessToken,
|
||||||
|
npmName: 'peertube-plugin-hello-world'
|
||||||
|
})
|
||||||
|
|
||||||
|
const settings = (res.body as RegisteredSettings).settings
|
||||||
|
|
||||||
|
expect(settings).to.have.length.at.least(1)
|
||||||
|
|
||||||
|
const adminNameSettings = settings.find(s => s.name === 'admin-name')
|
||||||
|
expect(adminNameSettings).to.not.be.undefined
|
||||||
})
|
})
|
||||||
|
|
||||||
it('Should update the settings', async function () {
|
it('Should update the settings', async function () {
|
||||||
// Update /settings
|
const settings = {
|
||||||
|
'admin-name': 'Cid'
|
||||||
|
}
|
||||||
|
|
||||||
// get /plugin
|
await updatePluginSettings({
|
||||||
|
url: server.url,
|
||||||
|
accessToken: server.accessToken,
|
||||||
|
npmName: 'peertube-plugin-hello-world',
|
||||||
|
settings
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should get a plugin and a theme', async function () {
|
||||||
|
{
|
||||||
|
const res = await getPlugin({
|
||||||
|
url: server.url,
|
||||||
|
accessToken: server.accessToken,
|
||||||
|
npmName: 'peertube-plugin-hello-world'
|
||||||
|
})
|
||||||
|
|
||||||
|
const plugin: PeerTubePlugin = res.body
|
||||||
|
|
||||||
|
expect(plugin.type).to.equal(PluginType.PLUGIN)
|
||||||
|
expect(plugin.name).to.equal('hello-world')
|
||||||
|
expect(plugin.description).to.exist
|
||||||
|
expect(plugin.homepage).to.exist
|
||||||
|
expect(plugin.uninstalled).to.be.false
|
||||||
|
expect(plugin.enabled).to.be.true
|
||||||
|
expect(plugin.description).to.exist
|
||||||
|
expect(plugin.version).to.exist
|
||||||
|
expect(plugin.peertubeEngine).to.exist
|
||||||
|
expect(plugin.createdAt).to.exist
|
||||||
|
|
||||||
|
expect(plugin.settings).to.not.be.undefined
|
||||||
|
expect(plugin.settings['admin-name']).to.equal('Cid')
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
const res = await getPlugin({
|
||||||
|
url: server.url,
|
||||||
|
accessToken: server.accessToken,
|
||||||
|
npmName: 'peertube-theme-background-red'
|
||||||
|
})
|
||||||
|
|
||||||
|
const plugin: PeerTubePlugin = res.body
|
||||||
|
|
||||||
|
expect(plugin.type).to.equal(PluginType.THEME)
|
||||||
|
expect(plugin.name).to.equal('background-red')
|
||||||
|
expect(plugin.description).to.exist
|
||||||
|
expect(plugin.homepage).to.exist
|
||||||
|
expect(plugin.uninstalled).to.be.false
|
||||||
|
expect(plugin.enabled).to.be.true
|
||||||
|
expect(plugin.description).to.exist
|
||||||
|
expect(plugin.version).to.exist
|
||||||
|
expect(plugin.peertubeEngine).to.exist
|
||||||
|
expect(plugin.createdAt).to.exist
|
||||||
|
|
||||||
|
expect(plugin.settings).to.be.null
|
||||||
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
it('Should update the plugin and the theme', async function () {
|
it('Should update the plugin and the theme', async function () {
|
||||||
// update BDD -> 0.0.1
|
this.timeout(30000)
|
||||||
// update package.json (theme + plugin)
|
|
||||||
// list to check versions
|
// Wait the scheduler that get the latest plugins versions
|
||||||
// update plugin + theme
|
await wait(6000)
|
||||||
// list to check they have been updated
|
|
||||||
// check package.json are upgraded too
|
// Fake update our plugin version
|
||||||
|
await setPluginVersion(server.internalServerNumber, 'hello-world', '0.0.1')
|
||||||
|
|
||||||
|
// Fake update package.json
|
||||||
|
const packageJSON: PluginPackageJson = await getPluginPackageJSON(server, 'peertube-plugin-hello-world')
|
||||||
|
const oldVersion = packageJSON.version
|
||||||
|
|
||||||
|
packageJSON.version = '0.0.1'
|
||||||
|
await updatePluginPackageJSON(server, 'peertube-plugin-hello-world', packageJSON)
|
||||||
|
|
||||||
|
// Restart the server to take into account this change
|
||||||
|
killallServers([ server ])
|
||||||
|
await reRunServer(server)
|
||||||
|
|
||||||
|
{
|
||||||
|
const res = await listPlugins({
|
||||||
|
url: server.url,
|
||||||
|
accessToken: server.accessToken,
|
||||||
|
pluginType: PluginType.PLUGIN
|
||||||
|
})
|
||||||
|
|
||||||
|
const plugin: PeerTubePlugin = res.body.data[0]
|
||||||
|
|
||||||
|
expect(plugin.version).to.equal('0.0.1')
|
||||||
|
expect(plugin.latestVersion).to.exist
|
||||||
|
expect(plugin.latestVersion).to.not.equal('0.0.1')
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
await updatePlugin({
|
||||||
|
url: server.url,
|
||||||
|
accessToken: server.accessToken,
|
||||||
|
npmName: 'peertube-plugin-hello-world'
|
||||||
|
})
|
||||||
|
|
||||||
|
const res = await listPlugins({
|
||||||
|
url: server.url,
|
||||||
|
accessToken: server.accessToken,
|
||||||
|
pluginType: PluginType.PLUGIN
|
||||||
|
})
|
||||||
|
|
||||||
|
const plugin: PeerTubePlugin = res.body.data[0]
|
||||||
|
|
||||||
|
expect(plugin.version).to.equal(oldVersion)
|
||||||
|
|
||||||
|
const updatedPackageJSON: PluginPackageJson = await getPluginPackageJSON(server, 'peertube-plugin-hello-world')
|
||||||
|
expect(updatedPackageJSON.version).to.equal(oldVersion)
|
||||||
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
it('Should uninstall the plugin', async function () {
|
it('Should uninstall the plugin', async function () {
|
||||||
// uninstall
|
await uninstallPlugin({
|
||||||
// list
|
url: server.url,
|
||||||
|
accessToken: server.accessToken,
|
||||||
|
npmName: 'peertube-plugin-hello-world'
|
||||||
|
})
|
||||||
|
|
||||||
|
const res = await listPlugins({
|
||||||
|
url: server.url,
|
||||||
|
accessToken: server.accessToken,
|
||||||
|
pluginType: PluginType.PLUGIN
|
||||||
|
})
|
||||||
|
|
||||||
|
expect(res.body.total).to.equal(0)
|
||||||
|
expect(res.body.data).to.have.lengthOf(0)
|
||||||
})
|
})
|
||||||
|
|
||||||
it('Should have an empty global css', async function () {
|
it('Should have an empty global css', async function () {
|
||||||
// get /global.css
|
const res = await getPluginsCSS(server.url)
|
||||||
|
|
||||||
|
expect(res.text).to.be.empty
|
||||||
})
|
})
|
||||||
|
|
||||||
it('Should list uninstalled plugins', async function () {
|
it('Should list uninstalled plugins', async function () {
|
||||||
// { uninstalled: true }
|
const res = await listPlugins({
|
||||||
|
url: server.url,
|
||||||
|
accessToken: server.accessToken,
|
||||||
|
pluginType: PluginType.PLUGIN,
|
||||||
|
uninstalled: true
|
||||||
|
})
|
||||||
|
|
||||||
|
expect(res.body.total).to.equal(1)
|
||||||
|
expect(res.body.data).to.have.lengthOf(1)
|
||||||
|
|
||||||
|
const plugin: PeerTubePlugin = res.body.data[0]
|
||||||
|
expect(plugin.name).to.equal('hello-world')
|
||||||
|
expect(plugin.enabled).to.be.false
|
||||||
|
expect(plugin.uninstalled).to.be.true
|
||||||
})
|
})
|
||||||
|
|
||||||
it('Should uninstall the theme', async function () {
|
it('Should uninstall the theme', async function () {
|
||||||
// Uninstall
|
await uninstallPlugin({
|
||||||
|
url: server.url,
|
||||||
|
accessToken: server.accessToken,
|
||||||
|
npmName: 'peertube-theme-background-red'
|
||||||
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
it('Should have updated the configuration', async function () {
|
it('Should have updated the configuration', async function () {
|
||||||
// get /config (default theme + registered themes + registered plugins)
|
// get /config (default theme + registered themes + registered plugins)
|
||||||
|
const res = await getConfig(server.url)
|
||||||
|
const config: ServerConfig = res.body
|
||||||
|
|
||||||
|
expect(config.theme.default).to.equal('default')
|
||||||
|
|
||||||
|
const theme = config.theme.registered.find(r => r.name === 'background-red')
|
||||||
|
expect(theme).to.be.undefined
|
||||||
|
|
||||||
|
const plugin = config.plugin.registered.find(r => r.name === 'hello-world')
|
||||||
|
expect(plugin).to.be.undefined
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should have updated the user theme', async function () {
|
||||||
|
const res = await getMyUserInformation(server.url, server.accessToken)
|
||||||
|
expect((res.body as User).theme).to.equal('instance-default')
|
||||||
})
|
})
|
||||||
|
|
||||||
after(async function () {
|
after(async function () {
|
||||||
|
await closeAllSequelize([ server ])
|
||||||
await cleanupTests([ server ])
|
await cleanupTests([ server ])
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
|
@ -7,7 +7,7 @@ import { setAccessTokensToServers } from '../../../shared/extra-utils'
|
||||||
|
|
||||||
const expect = chai.expect
|
const expect = chai.expect
|
||||||
|
|
||||||
describe('Test plugin filter hooks', function () {
|
describe('Test plugin action hooks', function () {
|
||||||
let server: ServerInfo
|
let server: ServerInfo
|
||||||
|
|
||||||
before(async function () {
|
before(async function () {
|
||||||
|
@ -18,7 +18,7 @@ describe('Test plugin filter hooks', function () {
|
||||||
})
|
})
|
||||||
|
|
||||||
it('Should execute ', async function () {
|
it('Should execute ', async function () {
|
||||||
|
// empty
|
||||||
})
|
})
|
||||||
|
|
||||||
after(async function () {
|
after(async function () {
|
||||||
|
|
|
@ -18,7 +18,7 @@ describe('Test plugin filter hooks', function () {
|
||||||
})
|
})
|
||||||
|
|
||||||
it('Should execute ', async function () {
|
it('Should execute ', async function () {
|
||||||
|
// empty
|
||||||
})
|
})
|
||||||
|
|
||||||
after(async function () {
|
after(async function () {
|
||||||
|
|
|
@ -65,9 +65,9 @@ async function pluginsListCLI () {
|
||||||
const { url, username, password } = await getServerCredentials(program)
|
const { url, username, password } = await getServerCredentials(program)
|
||||||
const accessToken = await getAdminTokenOrDie(url, username, password)
|
const accessToken = await getAdminTokenOrDie(url, username, password)
|
||||||
|
|
||||||
let type: PluginType
|
let pluginType: PluginType
|
||||||
if (program['onlyThemes']) type = PluginType.THEME
|
if (program['onlyThemes']) pluginType = PluginType.THEME
|
||||||
if (program['onlyPlugins']) type = PluginType.PLUGIN
|
if (program['onlyPlugins']) pluginType = PluginType.PLUGIN
|
||||||
|
|
||||||
const res = await listPlugins({
|
const res = await listPlugins({
|
||||||
url,
|
url,
|
||||||
|
@ -75,7 +75,7 @@ async function pluginsListCLI () {
|
||||||
start: 0,
|
start: 0,
|
||||||
count: 100,
|
count: 100,
|
||||||
sort: 'name',
|
sort: 'name',
|
||||||
type
|
pluginType
|
||||||
})
|
})
|
||||||
const plugins: PeerTubePlugin[] = res.body.data
|
const plugins: PeerTubePlugin[] = res.body.data
|
||||||
|
|
||||||
|
|
|
@ -1,5 +1,6 @@
|
||||||
import { QueryTypes, Sequelize } from 'sequelize'
|
import { QueryTypes, Sequelize } from 'sequelize'
|
||||||
import { ServerInfo } from '../server/servers'
|
import { ServerInfo } from '../server/servers'
|
||||||
|
import { PluginType } from '../../models/plugins/plugin.type'
|
||||||
|
|
||||||
let sequelizes: { [ id: number ]: Sequelize } = {}
|
let sequelizes: { [ id: number ]: Sequelize } = {}
|
||||||
|
|
||||||
|
@ -72,10 +73,19 @@ async function closeAllSequelize (servers: ServerInfo[]) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function setPluginVersion (internalServerNumber: number, pluginName: string, newVersion: string) {
|
||||||
|
const seq = getSequelize(internalServerNumber)
|
||||||
|
|
||||||
|
const options = { type: QueryTypes.UPDATE }
|
||||||
|
|
||||||
|
return seq.query(`UPDATE "plugin" SET "version" = '${newVersion}' WHERE "name" = '${pluginName}'`, options)
|
||||||
|
}
|
||||||
|
|
||||||
export {
|
export {
|
||||||
setVideoField,
|
setVideoField,
|
||||||
setPlaylistField,
|
setPlaylistField,
|
||||||
setActorField,
|
setActorField,
|
||||||
countVideoViewsOf,
|
countVideoViewsOf,
|
||||||
|
setPluginVersion,
|
||||||
closeAllSequelize
|
closeAllSequelize
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,5 +1,10 @@
|
||||||
import { makeGetRequest, makePostBodyRequest } from '../requests/requests'
|
import { makeGetRequest, makePostBodyRequest, makePutBodyRequest } from '../requests/requests'
|
||||||
import { PluginType } from '../../models/plugins/plugin.type'
|
import { PluginType } from '../../models/plugins/plugin.type'
|
||||||
|
import { PeertubePluginIndexList } from '../../models/plugins/peertube-plugin-index-list.model'
|
||||||
|
import { readJSON, writeJSON } from 'fs-extra'
|
||||||
|
import { ServerInfo } from './servers'
|
||||||
|
import { root } from '../miscs/miscs'
|
||||||
|
import { join } from 'path'
|
||||||
|
|
||||||
function listPlugins (parameters: {
|
function listPlugins (parameters: {
|
||||||
url: string,
|
url: string,
|
||||||
|
@ -7,10 +12,11 @@ function listPlugins (parameters: {
|
||||||
start?: number,
|
start?: number,
|
||||||
count?: number,
|
count?: number,
|
||||||
sort?: string,
|
sort?: string,
|
||||||
type?: PluginType,
|
pluginType?: PluginType,
|
||||||
|
uninstalled?: boolean,
|
||||||
expectedStatus?: number
|
expectedStatus?: number
|
||||||
}) {
|
}) {
|
||||||
const { url, accessToken, start, count, sort, type, expectedStatus = 200 } = parameters
|
const { url, accessToken, start, count, sort, pluginType, uninstalled, expectedStatus = 200 } = parameters
|
||||||
const path = '/api/v1/plugins'
|
const path = '/api/v1/plugins'
|
||||||
|
|
||||||
return makeGetRequest({
|
return makeGetRequest({
|
||||||
|
@ -21,12 +27,45 @@ function listPlugins (parameters: {
|
||||||
start,
|
start,
|
||||||
count,
|
count,
|
||||||
sort,
|
sort,
|
||||||
type
|
pluginType,
|
||||||
|
uninstalled
|
||||||
},
|
},
|
||||||
statusCodeExpected: expectedStatus
|
statusCodeExpected: expectedStatus
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function listAvailablePlugins (parameters: {
|
||||||
|
url: string,
|
||||||
|
accessToken: string,
|
||||||
|
start?: number,
|
||||||
|
count?: number,
|
||||||
|
sort?: string,
|
||||||
|
pluginType?: PluginType,
|
||||||
|
currentPeerTubeEngine?: string,
|
||||||
|
search?: string
|
||||||
|
expectedStatus?: number
|
||||||
|
}) {
|
||||||
|
const { url, accessToken, start, count, sort, pluginType, search, currentPeerTubeEngine, expectedStatus = 200 } = parameters
|
||||||
|
const path = '/api/v1/plugins/available'
|
||||||
|
|
||||||
|
const query: PeertubePluginIndexList = {
|
||||||
|
start,
|
||||||
|
count,
|
||||||
|
sort,
|
||||||
|
pluginType,
|
||||||
|
currentPeerTubeEngine,
|
||||||
|
search
|
||||||
|
}
|
||||||
|
|
||||||
|
return makeGetRequest({
|
||||||
|
url,
|
||||||
|
path,
|
||||||
|
token: accessToken,
|
||||||
|
query,
|
||||||
|
statusCodeExpected: expectedStatus
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
function getPlugin (parameters: {
|
function getPlugin (parameters: {
|
||||||
url: string,
|
url: string,
|
||||||
accessToken: string,
|
accessToken: string,
|
||||||
|
@ -44,19 +83,21 @@ function getPlugin (parameters: {
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
function getPluginSettings (parameters: {
|
function updatePluginSettings (parameters: {
|
||||||
url: string,
|
url: string,
|
||||||
accessToken: string,
|
accessToken: string,
|
||||||
npmName: string,
|
npmName: string,
|
||||||
|
settings: any,
|
||||||
expectedStatus?: number
|
expectedStatus?: number
|
||||||
}) {
|
}) {
|
||||||
const { url, accessToken, npmName, expectedStatus = 200 } = parameters
|
const { url, accessToken, npmName, settings, expectedStatus = 204 } = parameters
|
||||||
const path = '/api/v1/plugins/' + npmName + '/settings'
|
const path = '/api/v1/plugins/' + npmName + '/settings'
|
||||||
|
|
||||||
return makeGetRequest({
|
return makePutBodyRequest({
|
||||||
url,
|
url,
|
||||||
path,
|
path,
|
||||||
token: accessToken,
|
token: accessToken,
|
||||||
|
fields: { settings },
|
||||||
statusCodeExpected: expectedStatus
|
statusCodeExpected: expectedStatus
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
@ -134,12 +175,43 @@ function uninstallPlugin (parameters: {
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function getPluginsCSS (url: string) {
|
||||||
|
const path = '/plugins/global.css'
|
||||||
|
|
||||||
|
return makeGetRequest({
|
||||||
|
url,
|
||||||
|
path,
|
||||||
|
statusCodeExpected: 200
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
function getPackageJSONPath (server: ServerInfo, npmName: string) {
|
||||||
|
return join(root(), 'test' + server.internalServerNumber, 'plugins', 'node_modules', npmName, 'package.json')
|
||||||
|
}
|
||||||
|
|
||||||
|
function updatePluginPackageJSON (server: ServerInfo, npmName: string, json: any) {
|
||||||
|
const path = getPackageJSONPath(server, npmName)
|
||||||
|
|
||||||
|
return writeJSON(path, json)
|
||||||
|
}
|
||||||
|
|
||||||
|
function getPluginPackageJSON (server: ServerInfo, npmName: string) {
|
||||||
|
const path = getPackageJSONPath(server, npmName)
|
||||||
|
|
||||||
|
return readJSON(path)
|
||||||
|
}
|
||||||
|
|
||||||
export {
|
export {
|
||||||
listPlugins,
|
listPlugins,
|
||||||
|
listAvailablePlugins,
|
||||||
installPlugin,
|
installPlugin,
|
||||||
|
getPluginsCSS,
|
||||||
updatePlugin,
|
updatePlugin,
|
||||||
getPlugin,
|
getPlugin,
|
||||||
uninstallPlugin,
|
uninstallPlugin,
|
||||||
getPluginSettings,
|
updatePluginSettings,
|
||||||
getPluginRegisteredSettings
|
getPluginRegisteredSettings,
|
||||||
|
getPackageJSONPath,
|
||||||
|
updatePluginPackageJSON,
|
||||||
|
getPluginPackageJSON
|
||||||
}
|
}
|
||||||
|
|
|
@ -4,3 +4,7 @@ export interface RegisterSettingOptions {
|
||||||
type: 'input'
|
type: 'input'
|
||||||
default?: string
|
default?: string
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export interface RegisteredSettings {
|
||||||
|
settings: RegisterSettingOptions[]
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in New Issue