2024-05-23 11:42:14 +02:00
|
|
|
// SPDX-FileCopyrightText: 2024 John Livingston <https://www.john-livingston.fr/>
|
|
|
|
//
|
|
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
2023-08-08 18:26:40 +02:00
|
|
|
import type { RegisterServerOptions } from '@peertube/peertube-types'
|
|
|
|
import type { Router, Request, Response, NextFunction } from 'express'
|
2024-06-04 16:39:25 +02:00
|
|
|
import type { ChannelConfiguration, ChannelEmojisConfiguration, ChannelInfos } from '../../../../shared/lib/types'
|
2023-08-08 18:26:40 +02:00
|
|
|
import { asyncMiddleware } from '../../middlewares/async'
|
2023-09-06 15:23:39 +02:00
|
|
|
import { getCheckConfigurationChannelMiddleware } from '../../middlewares/configuration/channel'
|
2023-09-06 15:56:55 +02:00
|
|
|
import { checkConfigurationEnabledMiddleware } from '../../middlewares/configuration/configuration'
|
2023-09-18 12:23:35 +02:00
|
|
|
import {
|
|
|
|
getChannelConfigurationOptions,
|
|
|
|
getDefaultChannelConfigurationOptions,
|
|
|
|
storeChannelConfigurationOptions
|
|
|
|
} from '../../configuration/channel/storage'
|
2023-09-06 15:23:39 +02:00
|
|
|
import { sanitizeChannelConfigurationOptions } from '../../configuration/channel/sanitize'
|
2023-12-27 12:48:45 +01:00
|
|
|
import { getConverseJSParams } from '../../../lib/conversejs/params'
|
2024-06-04 16:39:25 +02:00
|
|
|
import { Emojis } from '../../../lib/emojis'
|
2024-09-05 18:28:54 +02:00
|
|
|
import { RoomChannel } from '../../../lib/room-channel'
|
|
|
|
import { updateProsodyRoom } from '../../../lib/prosody/api/manage-rooms'
|
2023-08-08 18:26:40 +02:00
|
|
|
|
2023-09-07 17:25:48 +02:00
|
|
|
async function initConfigurationApiRouter (options: RegisterServerOptions, router: Router): Promise<void> {
|
2023-08-08 18:26:40 +02:00
|
|
|
const logger = options.peertubeHelpers.logger
|
|
|
|
|
2023-12-27 12:48:45 +01:00
|
|
|
router.get('/configuration/room/:roomKey', asyncMiddleware(
|
|
|
|
async (req: Request, res: Response, _next: NextFunction): Promise<void> => {
|
|
|
|
const roomKey = req.params.roomKey
|
2024-04-04 10:58:16 +02:00
|
|
|
|
|
|
|
const user = await options.peertubeHelpers.user.getAuthUser(res)
|
|
|
|
|
|
|
|
const initConverseJSParam = await getConverseJSParams(
|
|
|
|
options,
|
|
|
|
roomKey,
|
|
|
|
{
|
|
|
|
forcetype: req.query.forcetype === '1'
|
|
|
|
},
|
|
|
|
!!user
|
|
|
|
)
|
2023-12-27 12:48:45 +01:00
|
|
|
if (('isError' in initConverseJSParam) && initConverseJSParam.isError) {
|
|
|
|
res.sendStatus(initConverseJSParam.code)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
res.status(200)
|
|
|
|
res.json(initConverseJSParam)
|
|
|
|
}
|
|
|
|
))
|
|
|
|
|
2023-09-07 17:25:48 +02:00
|
|
|
router.get('/configuration/channel/:channelId', asyncMiddleware([
|
2023-09-06 15:56:55 +02:00
|
|
|
checkConfigurationEnabledMiddleware(options),
|
2023-09-06 15:23:39 +02:00
|
|
|
getCheckConfigurationChannelMiddleware(options),
|
2023-08-08 18:26:40 +02:00
|
|
|
async (req: Request, res: Response, _next: NextFunction): Promise<void> => {
|
2023-08-09 16:16:02 +02:00
|
|
|
if (!res.locals.channelInfos) {
|
|
|
|
logger.error('Missing channelInfos in res.locals, should not happen')
|
|
|
|
res.sendStatus(500)
|
2023-08-08 18:26:40 +02:00
|
|
|
return
|
|
|
|
}
|
2023-08-09 16:16:02 +02:00
|
|
|
const channelInfos = res.locals.channelInfos as ChannelInfos
|
|
|
|
|
2023-09-18 12:23:35 +02:00
|
|
|
const channelOptions =
|
|
|
|
await getChannelConfigurationOptions(options, channelInfos.id) ??
|
|
|
|
getDefaultChannelConfigurationOptions(options)
|
|
|
|
|
2024-06-04 16:39:25 +02:00
|
|
|
const result: ChannelConfiguration = {
|
2023-09-18 12:23:35 +02:00
|
|
|
channel: channelInfos,
|
|
|
|
configuration: channelOptions
|
|
|
|
}
|
2023-08-09 16:16:02 +02:00
|
|
|
res.status(200)
|
|
|
|
res.json(result)
|
|
|
|
}
|
|
|
|
]))
|
2023-08-08 18:26:40 +02:00
|
|
|
|
2023-09-07 17:25:48 +02:00
|
|
|
router.post('/configuration/channel/:channelId', asyncMiddleware([
|
2023-09-06 15:56:55 +02:00
|
|
|
checkConfigurationEnabledMiddleware(options),
|
2023-09-06 15:23:39 +02:00
|
|
|
getCheckConfigurationChannelMiddleware(options),
|
2023-08-09 16:16:02 +02:00
|
|
|
async (req: Request, res: Response, _next: NextFunction): Promise<void> => {
|
|
|
|
if (!res.locals.channelInfos) {
|
|
|
|
logger.error('Missing channelInfos in res.locals, should not happen')
|
|
|
|
res.sendStatus(500)
|
2023-08-08 18:26:40 +02:00
|
|
|
return
|
|
|
|
}
|
2023-08-09 16:16:02 +02:00
|
|
|
const channelInfos = res.locals.channelInfos as ChannelInfos
|
2023-09-06 15:23:39 +02:00
|
|
|
logger.debug('Trying to save ChannelConfigurationOptions')
|
2023-08-08 18:26:40 +02:00
|
|
|
|
2023-09-18 12:23:35 +02:00
|
|
|
let channelOptions
|
2023-08-09 16:16:02 +02:00
|
|
|
try {
|
2023-09-21 17:13:28 +02:00
|
|
|
// Note: the front-end should do some input validation.
|
|
|
|
// If there is any invalid value, we just return a 400 error.
|
|
|
|
// The frontend should have prevented to post invalid data.
|
2023-09-21 19:32:47 +02:00
|
|
|
|
|
|
|
// Note: if !bot.enabled, we wont try to save hidden fields values, to minimize the risk of error
|
|
|
|
if (req.body.bot?.enabled === false) {
|
|
|
|
logger.debug('Bot disabled, loading the previous bot conf to not override hidden fields')
|
|
|
|
const channelOptions =
|
|
|
|
await getChannelConfigurationOptions(options, channelInfos.id) ??
|
|
|
|
getDefaultChannelConfigurationOptions(options)
|
|
|
|
req.body.bot = channelOptions.bot
|
2023-09-26 14:37:56 +02:00
|
|
|
req.body.bot.enabled = false
|
2023-09-21 19:32:47 +02:00
|
|
|
}
|
2024-09-10 18:59:56 +02:00
|
|
|
// TODO: Same for forbidSpecialChars/noDuplicate: if disabled, don't save reason and tolerance
|
2024-09-07 00:46:23 +02:00
|
|
|
// (disabling for now, because it is not acceptable to load twice the channel configuration.
|
|
|
|
// Must find better way)
|
|
|
|
// if (req.body.bot?.enabled === true && req.body.bot.forbidSpecialChars?.enabled === false) {
|
|
|
|
// logger.debug('Bot disabled, loading the previous bot conf to not override hidden fields')
|
|
|
|
// const channelOptions =
|
|
|
|
// await getChannelConfigurationOptions(options, channelInfos.id) ??
|
|
|
|
// getDefaultChannelConfigurationOptions(options)
|
|
|
|
// req.body.bot.forbidSpecialChars.reason = channelOptions.bot.forbidSpecialChars.reason
|
|
|
|
// req.body.bot.forbidSpecialChars.tolerance = channelOptions.bot.forbidSpecialChars.tolerance
|
|
|
|
// req.body.bot.forbidSpecialChars.applyToModerators = channelOptions.bot.forbidSpecialChars.applyToModerators
|
|
|
|
// req.body.bot.forbidSpecialChars.enabled = false
|
2024-09-10 18:59:56 +02:00
|
|
|
// ... NoDuplicate...
|
2024-09-07 00:46:23 +02:00
|
|
|
// }
|
2023-09-18 12:23:35 +02:00
|
|
|
channelOptions = await sanitizeChannelConfigurationOptions(options, channelInfos.id, req.body)
|
2023-08-09 16:16:02 +02:00
|
|
|
} catch (err) {
|
|
|
|
logger.warn(err)
|
|
|
|
res.sendStatus(400)
|
2023-08-08 18:26:40 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2023-08-09 16:16:02 +02:00
|
|
|
logger.debug('Data seems ok, storing them.')
|
2024-06-04 16:39:25 +02:00
|
|
|
const result: ChannelConfiguration = {
|
2023-08-09 16:16:02 +02:00
|
|
|
channel: channelInfos,
|
2023-09-18 12:23:35 +02:00
|
|
|
configuration: channelOptions
|
2023-08-09 16:39:12 +02:00
|
|
|
}
|
2023-09-18 12:23:35 +02:00
|
|
|
await storeChannelConfigurationOptions(options, channelInfos.id, channelOptions)
|
2023-08-08 18:26:40 +02:00
|
|
|
res.status(200)
|
|
|
|
res.json(result)
|
|
|
|
}
|
2023-08-09 16:16:02 +02:00
|
|
|
]))
|
2024-05-28 17:56:24 +02:00
|
|
|
|
|
|
|
router.get('/configuration/channel/emojis/:channelId', asyncMiddleware([
|
|
|
|
checkConfigurationEnabledMiddleware(options),
|
2024-06-04 16:39:25 +02:00
|
|
|
getCheckConfigurationChannelMiddleware(options),
|
2024-05-28 17:56:24 +02:00
|
|
|
async (req: Request, res: Response, _next: NextFunction): Promise<void> => {
|
2024-06-04 16:39:25 +02:00
|
|
|
try {
|
|
|
|
if (!res.locals.channelInfos) {
|
|
|
|
throw new Error('Missing channelInfos in res.locals, should not happen')
|
|
|
|
}
|
|
|
|
|
|
|
|
const emojis = Emojis.singleton()
|
|
|
|
const channelInfos = res.locals.channelInfos as ChannelInfos
|
|
|
|
|
|
|
|
const channelEmojis =
|
|
|
|
(await emojis.channelCustomEmojisDefinition(channelInfos.id)) ??
|
|
|
|
emojis.emptyChannelDefinition()
|
|
|
|
|
|
|
|
const result: ChannelEmojisConfiguration = {
|
|
|
|
channel: channelInfos,
|
|
|
|
emojis: channelEmojis
|
|
|
|
}
|
|
|
|
res.status(200)
|
|
|
|
res.json(result)
|
|
|
|
} catch (err) {
|
|
|
|
logger.error(err)
|
2024-05-28 17:56:24 +02:00
|
|
|
res.sendStatus(500)
|
|
|
|
}
|
2024-06-04 16:39:25 +02:00
|
|
|
}
|
|
|
|
]))
|
2024-05-28 17:56:24 +02:00
|
|
|
|
2024-06-04 16:39:25 +02:00
|
|
|
router.post('/configuration/channel/emojis/:channelId', asyncMiddleware([
|
|
|
|
checkConfigurationEnabledMiddleware(options),
|
|
|
|
getCheckConfigurationChannelMiddleware(options),
|
|
|
|
async (req: Request, res: Response, _next: NextFunction): Promise<void> => {
|
|
|
|
try {
|
|
|
|
if (!res.locals.channelInfos) {
|
|
|
|
throw new Error('Missing channelInfos in res.locals, should not happen')
|
|
|
|
}
|
|
|
|
|
|
|
|
const emojis = Emojis.singleton()
|
|
|
|
const channelInfos = res.locals.channelInfos as ChannelInfos
|
|
|
|
|
|
|
|
const emojisDefinition = req.body
|
2024-06-06 11:36:07 +02:00
|
|
|
let emojisDefinitionSanitized, bufferInfos
|
2024-06-04 16:39:25 +02:00
|
|
|
try {
|
2024-06-06 11:36:07 +02:00
|
|
|
[emojisDefinitionSanitized, bufferInfos] = await emojis.sanitizeChannelDefinition(
|
|
|
|
channelInfos.id,
|
|
|
|
emojisDefinition
|
|
|
|
)
|
2024-06-04 16:39:25 +02:00
|
|
|
} catch (err) {
|
|
|
|
logger.warn(err)
|
|
|
|
res.sendStatus(400)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2024-06-06 11:36:07 +02:00
|
|
|
await emojis.saveChannelDefinition(channelInfos.id, emojisDefinitionSanitized, bufferInfos)
|
2024-06-04 16:39:25 +02:00
|
|
|
|
2024-09-05 18:28:54 +02:00
|
|
|
// We must update the emoji only regexp on the Prosody server.
|
2024-09-06 11:01:48 +02:00
|
|
|
const customEmojisRegexp = await emojis.getChannelCustomEmojisRegexp(channelInfos.id)
|
2024-09-05 18:28:54 +02:00
|
|
|
const roomJIDs = RoomChannel.singleton().getChannelRoomJIDs(channelInfos.id)
|
|
|
|
for (const roomJID of roomJIDs) {
|
|
|
|
// No need to await here
|
|
|
|
logger.info(`Updating room ${roomJID} emoji only regexp...`)
|
|
|
|
updateProsodyRoom(options, roomJID, {
|
2024-09-06 11:01:48 +02:00
|
|
|
livechat_custom_emoji_regexp: customEmojisRegexp
|
2024-09-05 18:28:54 +02:00
|
|
|
}).then(
|
|
|
|
() => {},
|
|
|
|
(err) => logger.error(err)
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2024-06-20 11:14:00 +02:00
|
|
|
// Reloading data, to send them back to front:
|
|
|
|
const channelEmojis =
|
|
|
|
(await emojis.channelCustomEmojisDefinition(channelInfos.id)) ??
|
|
|
|
emojis.emptyChannelDefinition()
|
|
|
|
const result: ChannelEmojisConfiguration = {
|
|
|
|
channel: channelInfos,
|
|
|
|
emojis: channelEmojis
|
|
|
|
}
|
|
|
|
res.status(200)
|
|
|
|
res.json(result)
|
2024-06-04 16:39:25 +02:00
|
|
|
} catch (err) {
|
|
|
|
logger.error(err)
|
|
|
|
res.sendStatus(500)
|
|
|
|
}
|
2024-05-28 17:56:24 +02:00
|
|
|
}
|
|
|
|
]))
|
2024-09-06 11:53:07 +02:00
|
|
|
|
|
|
|
router.post('/configuration/channel/emojis/:channelId/enable_emoji_only', asyncMiddleware([
|
|
|
|
checkConfigurationEnabledMiddleware(options),
|
|
|
|
getCheckConfigurationChannelMiddleware(options),
|
|
|
|
async (req: Request, res: Response, _next: NextFunction): Promise<void> => {
|
|
|
|
try {
|
|
|
|
if (!res.locals.channelInfos) {
|
|
|
|
throw new Error('Missing channelInfos in res.locals, should not happen')
|
|
|
|
}
|
|
|
|
|
|
|
|
const emojis = Emojis.singleton()
|
|
|
|
const channelInfos = res.locals.channelInfos as ChannelInfos
|
|
|
|
|
|
|
|
logger.info(`Enabling emoji only mode on each channel ${channelInfos.id} rooms ...`)
|
|
|
|
|
|
|
|
// We can also update the EmojisRegexp, just in case.
|
|
|
|
const customEmojisRegexp = await emojis.getChannelCustomEmojisRegexp(channelInfos.id)
|
|
|
|
const roomJIDs = RoomChannel.singleton().getChannelRoomJIDs(channelInfos.id)
|
|
|
|
for (const roomJID of roomJIDs) {
|
|
|
|
// No need to await here
|
|
|
|
logger.info(`Enabling emoji only mode on room ${roomJID} ...`)
|
|
|
|
updateProsodyRoom(options, roomJID, {
|
|
|
|
livechat_emoji_only: true,
|
|
|
|
livechat_custom_emoji_regexp: customEmojisRegexp
|
|
|
|
}).then(
|
|
|
|
() => {},
|
|
|
|
(err) => logger.error(err)
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
res.status(200)
|
|
|
|
res.json({ ok: true })
|
|
|
|
} catch (err) {
|
|
|
|
logger.error(err)
|
|
|
|
res.sendStatus(500)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
]))
|
2023-08-08 18:26:40 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
export {
|
2023-09-06 15:23:39 +02:00
|
|
|
initConfigurationApiRouter
|
2023-08-08 18:26:40 +02:00
|
|
|
}
|