2018-04-24 10:05:32 -05:00
|
|
|
import * as express from 'express'
|
2020-12-08 03:30:33 -06:00
|
|
|
import { Hooks } from '@server/lib/plugins/hooks'
|
2020-08-20 02:19:21 -05:00
|
|
|
import { getServerActor } from '@server/models/application/application'
|
|
|
|
import { MChannelAccountDefault } from '@server/types/models'
|
|
|
|
import { VideoChannelCreate, VideoChannelUpdate } from '../../../shared'
|
2021-02-26 07:22:25 -06:00
|
|
|
import { HttpStatusCode } from '../../../shared/core-utils/miscs/http-error-codes'
|
2020-08-20 02:19:21 -05:00
|
|
|
import { auditLoggerFactory, getAuditIdFromRes, VideoChannelAuditView } from '../../helpers/audit-logger'
|
|
|
|
import { resetSequelizeInstance } from '../../helpers/database-utils'
|
|
|
|
import { buildNSFWFilter, createReqFiles, getCountVideos, isUserAbleToSearchRemoteURI } from '../../helpers/express-utils'
|
|
|
|
import { logger } from '../../helpers/logger'
|
2020-04-23 04:36:50 -05:00
|
|
|
import { getFormattedObjects } from '../../helpers/utils'
|
2020-08-20 02:19:21 -05:00
|
|
|
import { CONFIG } from '../../initializers/config'
|
|
|
|
import { MIMETYPES } from '../../initializers/constants'
|
|
|
|
import { sequelizeTypescript } from '../../initializers/database'
|
|
|
|
import { sendUpdateActor } from '../../lib/activitypub/send'
|
2021-04-06 04:35:56 -05:00
|
|
|
import { deleteLocalActorAvatarFile, updateLocalActorAvatarFile } from '../../lib/actor-image'
|
2020-08-20 02:19:21 -05:00
|
|
|
import { JobQueue } from '../../lib/job-queue'
|
|
|
|
import { createLocalVideoChannel, federateAllVideosOfChannel } from '../../lib/video-channel'
|
2018-04-24 10:05:32 -05:00
|
|
|
import {
|
|
|
|
asyncMiddleware,
|
2018-06-13 07:27:40 -05:00
|
|
|
asyncRetryTransactionMiddleware,
|
2018-08-14 08:28:30 -05:00
|
|
|
authenticate,
|
|
|
|
commonVideosFiltersValidator,
|
2018-04-25 09:15:39 -05:00
|
|
|
optionalAuthenticate,
|
2018-04-24 10:05:32 -05:00
|
|
|
paginationValidator,
|
|
|
|
setDefaultPagination,
|
|
|
|
setDefaultSort,
|
2020-08-20 02:19:21 -05:00
|
|
|
setDefaultVideosSort,
|
2018-04-25 09:15:39 -05:00
|
|
|
videoChannelsAddValidator,
|
|
|
|
videoChannelsRemoveValidator,
|
|
|
|
videoChannelsSortValidator,
|
2019-02-26 03:55:40 -06:00
|
|
|
videoChannelsUpdateValidator,
|
|
|
|
videoPlaylistsSortValidator
|
2018-04-24 10:05:32 -05:00
|
|
|
} from '../../middlewares'
|
2020-08-20 02:19:21 -05:00
|
|
|
import { videoChannelsNameWithHostValidator, videoChannelsOwnSearchValidator, videosSortValidator } from '../../middlewares/validators'
|
|
|
|
import { updateAvatarValidator } from '../../middlewares/validators/avatar'
|
|
|
|
import { commonVideoPlaylistFiltersValidator } from '../../middlewares/validators/videos/video-playlists'
|
2018-04-25 09:15:39 -05:00
|
|
|
import { AccountModel } from '../../models/account/account'
|
|
|
|
import { VideoModel } from '../../models/video/video'
|
2020-08-20 02:19:21 -05:00
|
|
|
import { VideoChannelModel } from '../../models/video/video-channel'
|
2019-02-26 03:55:40 -06:00
|
|
|
import { VideoPlaylistModel } from '../../models/video/video-playlist'
|
2018-06-29 04:29:23 -05:00
|
|
|
|
2018-07-31 07:04:26 -05:00
|
|
|
const auditLogger = auditLoggerFactory('channels')
|
2018-12-11 07:52:50 -06:00
|
|
|
const reqAvatarFile = createReqFiles([ 'avatarfile' ], MIMETYPES.IMAGE.MIMETYPE_EXT, { avatarfile: CONFIG.STORAGE.TMP_DIR })
|
2018-04-24 10:05:32 -05:00
|
|
|
|
|
|
|
const videoChannelRouter = express.Router()
|
|
|
|
|
|
|
|
videoChannelRouter.get('/',
|
|
|
|
paginationValidator,
|
|
|
|
videoChannelsSortValidator,
|
|
|
|
setDefaultSort,
|
|
|
|
setDefaultPagination,
|
2020-07-15 04:17:03 -05:00
|
|
|
videoChannelsOwnSearchValidator,
|
2018-04-24 10:05:32 -05:00
|
|
|
asyncMiddleware(listVideoChannels)
|
|
|
|
)
|
|
|
|
|
2018-04-25 09:15:39 -05:00
|
|
|
videoChannelRouter.post('/',
|
|
|
|
authenticate,
|
2018-10-01 08:18:07 -05:00
|
|
|
asyncMiddleware(videoChannelsAddValidator),
|
2018-06-13 07:27:40 -05:00
|
|
|
asyncRetryTransactionMiddleware(addVideoChannel)
|
2018-04-25 09:15:39 -05:00
|
|
|
)
|
|
|
|
|
2018-08-17 08:45:42 -05:00
|
|
|
videoChannelRouter.post('/:nameWithHost/avatar/pick',
|
2018-06-29 04:29:23 -05:00
|
|
|
authenticate,
|
|
|
|
reqAvatarFile,
|
|
|
|
// Check the rights
|
|
|
|
asyncMiddleware(videoChannelsUpdateValidator),
|
|
|
|
updateAvatarValidator,
|
2018-09-26 03:15:50 -05:00
|
|
|
asyncMiddleware(updateVideoChannelAvatar)
|
2018-06-29 04:29:23 -05:00
|
|
|
)
|
|
|
|
|
2021-01-13 02:12:55 -06:00
|
|
|
videoChannelRouter.delete('/:nameWithHost/avatar',
|
|
|
|
authenticate,
|
|
|
|
// Check the rights
|
|
|
|
asyncMiddleware(videoChannelsUpdateValidator),
|
|
|
|
asyncMiddleware(deleteVideoChannelAvatar)
|
|
|
|
)
|
|
|
|
|
2018-08-17 08:45:42 -05:00
|
|
|
videoChannelRouter.put('/:nameWithHost',
|
2018-04-25 09:15:39 -05:00
|
|
|
authenticate,
|
|
|
|
asyncMiddleware(videoChannelsUpdateValidator),
|
2018-06-13 07:27:40 -05:00
|
|
|
asyncRetryTransactionMiddleware(updateVideoChannel)
|
2018-04-25 09:15:39 -05:00
|
|
|
)
|
|
|
|
|
2018-08-17 08:45:42 -05:00
|
|
|
videoChannelRouter.delete('/:nameWithHost',
|
2018-04-25 09:15:39 -05:00
|
|
|
authenticate,
|
|
|
|
asyncMiddleware(videoChannelsRemoveValidator),
|
2018-06-13 07:27:40 -05:00
|
|
|
asyncRetryTransactionMiddleware(removeVideoChannel)
|
2018-04-25 09:15:39 -05:00
|
|
|
)
|
|
|
|
|
2018-08-17 08:45:42 -05:00
|
|
|
videoChannelRouter.get('/:nameWithHost',
|
|
|
|
asyncMiddleware(videoChannelsNameWithHostValidator),
|
2018-04-25 09:15:39 -05:00
|
|
|
asyncMiddleware(getVideoChannel)
|
|
|
|
)
|
|
|
|
|
2019-02-26 03:55:40 -06:00
|
|
|
videoChannelRouter.get('/:nameWithHost/video-playlists',
|
|
|
|
asyncMiddleware(videoChannelsNameWithHostValidator),
|
|
|
|
paginationValidator,
|
|
|
|
videoPlaylistsSortValidator,
|
|
|
|
setDefaultSort,
|
|
|
|
setDefaultPagination,
|
2019-03-05 03:58:44 -06:00
|
|
|
commonVideoPlaylistFiltersValidator,
|
2019-02-26 03:55:40 -06:00
|
|
|
asyncMiddleware(listVideoChannelPlaylists)
|
|
|
|
)
|
|
|
|
|
2018-08-17 08:45:42 -05:00
|
|
|
videoChannelRouter.get('/:nameWithHost/videos',
|
|
|
|
asyncMiddleware(videoChannelsNameWithHostValidator),
|
2018-04-25 09:15:39 -05:00
|
|
|
paginationValidator,
|
|
|
|
videosSortValidator,
|
2020-08-20 02:19:21 -05:00
|
|
|
setDefaultVideosSort,
|
2018-04-25 09:15:39 -05:00
|
|
|
setDefaultPagination,
|
|
|
|
optionalAuthenticate,
|
2018-07-20 07:35:18 -05:00
|
|
|
commonVideosFiltersValidator,
|
2018-04-25 09:15:39 -05:00
|
|
|
asyncMiddleware(listVideoChannelVideos)
|
|
|
|
)
|
|
|
|
|
2018-04-24 10:05:32 -05:00
|
|
|
// ---------------------------------------------------------------------------
|
|
|
|
|
|
|
|
export {
|
|
|
|
videoChannelRouter
|
|
|
|
}
|
|
|
|
|
|
|
|
// ---------------------------------------------------------------------------
|
|
|
|
|
2019-03-19 04:35:15 -05:00
|
|
|
async function listVideoChannels (req: express.Request, res: express.Response) {
|
2018-08-23 10:58:39 -05:00
|
|
|
const serverActor = await getServerActor()
|
2020-07-15 04:17:03 -05:00
|
|
|
const resultList = await VideoChannelModel.listForApi({
|
|
|
|
actorId: serverActor.id,
|
|
|
|
start: req.query.start,
|
|
|
|
count: req.query.count,
|
2020-07-23 14:30:04 -05:00
|
|
|
sort: req.query.sort
|
2020-07-15 04:17:03 -05:00
|
|
|
})
|
2018-04-24 10:05:32 -05:00
|
|
|
|
|
|
|
return res.json(getFormattedObjects(resultList.data, resultList.total))
|
|
|
|
}
|
2018-04-25 09:15:39 -05:00
|
|
|
|
2019-03-19 04:35:15 -05:00
|
|
|
async function updateVideoChannelAvatar (req: express.Request, res: express.Response) {
|
2020-01-31 09:56:52 -06:00
|
|
|
const avatarPhysicalFile = req.files['avatarfile'][0]
|
2019-03-19 04:35:15 -05:00
|
|
|
const videoChannel = res.locals.videoChannel
|
2018-07-31 07:04:26 -05:00
|
|
|
const oldVideoChannelAuditKeys = new VideoChannelAuditView(videoChannel.toFormattedJSON())
|
2018-06-29 04:29:23 -05:00
|
|
|
|
2021-01-26 03:23:21 -06:00
|
|
|
const avatar = await updateLocalActorAvatarFile(videoChannel, avatarPhysicalFile)
|
2018-06-29 04:29:23 -05:00
|
|
|
|
2018-09-20 04:31:48 -05:00
|
|
|
auditLogger.update(getAuditIdFromRes(res), new VideoChannelAuditView(videoChannel.toFormattedJSON()), oldVideoChannelAuditKeys)
|
2018-07-31 07:04:26 -05:00
|
|
|
|
2018-06-29 04:29:23 -05:00
|
|
|
return res
|
|
|
|
.json({
|
|
|
|
avatar: avatar.toFormattedJSON()
|
|
|
|
})
|
|
|
|
.end()
|
|
|
|
}
|
|
|
|
|
2021-01-13 02:12:55 -06:00
|
|
|
async function deleteVideoChannelAvatar (req: express.Request, res: express.Response) {
|
|
|
|
const videoChannel = res.locals.videoChannel
|
|
|
|
|
2021-01-26 03:23:21 -06:00
|
|
|
await deleteLocalActorAvatarFile(videoChannel)
|
2021-01-13 02:12:55 -06:00
|
|
|
|
|
|
|
return res.sendStatus(HttpStatusCode.NO_CONTENT_204)
|
|
|
|
}
|
|
|
|
|
2018-04-25 09:15:39 -05:00
|
|
|
async function addVideoChannel (req: express.Request, res: express.Response) {
|
|
|
|
const videoChannelInfo: VideoChannelCreate = req.body
|
|
|
|
|
2019-08-15 04:53:26 -05:00
|
|
|
const videoChannelCreated = await sequelizeTypescript.transaction(async t => {
|
2019-03-19 04:35:15 -05:00
|
|
|
const account = await AccountModel.load(res.locals.oauth.token.User.Account.id, t)
|
2018-09-20 03:13:13 -05:00
|
|
|
|
2019-08-20 12:05:31 -05:00
|
|
|
return createLocalVideoChannel(videoChannelInfo, account, t)
|
2018-04-25 09:15:39 -05:00
|
|
|
})
|
|
|
|
|
2021-02-26 07:22:25 -06:00
|
|
|
const payload = { actorId: videoChannelCreated.actorId }
|
|
|
|
await JobQueue.Instance.createJobWithPromise({ type: 'actor-keys', payload })
|
2018-04-25 09:15:39 -05:00
|
|
|
|
2018-09-20 03:13:13 -05:00
|
|
|
auditLogger.create(getAuditIdFromRes(res), new VideoChannelAuditView(videoChannelCreated.toFormattedJSON()))
|
2019-05-31 07:02:26 -05:00
|
|
|
logger.info('Video channel %s created.', videoChannelCreated.Actor.url)
|
2018-04-25 09:15:39 -05:00
|
|
|
|
2018-06-13 07:27:40 -05:00
|
|
|
return res.json({
|
|
|
|
videoChannel: {
|
2019-05-31 07:02:26 -05:00
|
|
|
id: videoChannelCreated.id
|
2018-06-13 07:27:40 -05:00
|
|
|
}
|
|
|
|
}).end()
|
2018-04-25 09:15:39 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
async function updateVideoChannel (req: express.Request, res: express.Response) {
|
2019-03-19 04:35:15 -05:00
|
|
|
const videoChannelInstance = res.locals.videoChannel
|
2018-04-25 09:15:39 -05:00
|
|
|
const videoChannelFieldsSave = videoChannelInstance.toJSON()
|
2018-07-31 07:04:26 -05:00
|
|
|
const oldVideoChannelAuditKeys = new VideoChannelAuditView(videoChannelInstance.toFormattedJSON())
|
2018-04-25 09:15:39 -05:00
|
|
|
const videoChannelInfoToUpdate = req.body as VideoChannelUpdate
|
2019-05-31 09:30:11 -05:00
|
|
|
let doBulkVideoUpdate = false
|
2018-04-25 09:15:39 -05:00
|
|
|
|
|
|
|
try {
|
|
|
|
await sequelizeTypescript.transaction(async t => {
|
|
|
|
const sequelizeOptions = {
|
|
|
|
transaction: t
|
|
|
|
}
|
|
|
|
|
2019-05-31 09:30:11 -05:00
|
|
|
if (videoChannelInfoToUpdate.displayName !== undefined) videoChannelInstance.name = videoChannelInfoToUpdate.displayName
|
|
|
|
if (videoChannelInfoToUpdate.description !== undefined) videoChannelInstance.description = videoChannelInfoToUpdate.description
|
|
|
|
|
|
|
|
if (videoChannelInfoToUpdate.support !== undefined) {
|
|
|
|
const oldSupportField = videoChannelInstance.support
|
|
|
|
videoChannelInstance.support = videoChannelInfoToUpdate.support
|
|
|
|
|
|
|
|
if (videoChannelInfoToUpdate.bulkVideosSupportUpdate === true && oldSupportField !== videoChannelInfoToUpdate.support) {
|
|
|
|
doBulkVideoUpdate = true
|
|
|
|
await VideoModel.bulkUpdateSupportField(videoChannelInstance, t)
|
|
|
|
}
|
|
|
|
}
|
2018-04-25 09:15:39 -05:00
|
|
|
|
2019-08-21 07:31:57 -05:00
|
|
|
const videoChannelInstanceUpdated = await videoChannelInstance.save(sequelizeOptions) as MChannelAccountDefault
|
2018-04-25 09:15:39 -05:00
|
|
|
await sendUpdateActor(videoChannelInstanceUpdated, t)
|
|
|
|
|
2018-07-31 07:04:26 -05:00
|
|
|
auditLogger.update(
|
2018-09-19 10:02:16 -05:00
|
|
|
getAuditIdFromRes(res),
|
2018-07-31 07:04:26 -05:00
|
|
|
new VideoChannelAuditView(videoChannelInstanceUpdated.toFormattedJSON()),
|
|
|
|
oldVideoChannelAuditKeys
|
|
|
|
)
|
2019-05-31 09:30:11 -05:00
|
|
|
|
2019-05-31 07:02:26 -05:00
|
|
|
logger.info('Video channel %s updated.', videoChannelInstance.Actor.url)
|
2018-07-31 07:04:26 -05:00
|
|
|
})
|
2018-04-25 09:15:39 -05:00
|
|
|
} catch (err) {
|
|
|
|
logger.debug('Cannot update the video channel.', { err })
|
|
|
|
|
|
|
|
// Force fields we want to update
|
|
|
|
// If the transaction is retried, sequelize will think the object has not changed
|
|
|
|
// So it will skip the SQL request, even if the last one was ROLLBACKed!
|
|
|
|
resetSequelizeInstance(videoChannelInstance, videoChannelFieldsSave)
|
|
|
|
|
|
|
|
throw err
|
|
|
|
}
|
|
|
|
|
2020-12-07 07:32:36 -06:00
|
|
|
res.type('json').status(HttpStatusCode.NO_CONTENT_204).end()
|
2019-05-31 09:30:11 -05:00
|
|
|
|
|
|
|
// Don't process in a transaction, and after the response because it could be long
|
|
|
|
if (doBulkVideoUpdate) {
|
|
|
|
await federateAllVideosOfChannel(videoChannelInstance)
|
|
|
|
}
|
2018-04-25 09:15:39 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
async function removeVideoChannel (req: express.Request, res: express.Response) {
|
2019-03-19 04:35:15 -05:00
|
|
|
const videoChannelInstance = res.locals.videoChannel
|
2018-04-25 09:15:39 -05:00
|
|
|
|
2018-06-13 07:27:40 -05:00
|
|
|
await sequelizeTypescript.transaction(async t => {
|
2019-03-05 03:58:44 -06:00
|
|
|
await VideoPlaylistModel.resetPlaylistsOfChannel(videoChannelInstance.id, t)
|
|
|
|
|
2018-04-25 09:15:39 -05:00
|
|
|
await videoChannelInstance.destroy({ transaction: t })
|
|
|
|
|
2018-09-19 10:02:16 -05:00
|
|
|
auditLogger.delete(getAuditIdFromRes(res), new VideoChannelAuditView(videoChannelInstance.toFormattedJSON()))
|
2019-05-31 07:02:26 -05:00
|
|
|
logger.info('Video channel %s deleted.', videoChannelInstance.Actor.url)
|
2018-04-25 09:15:39 -05:00
|
|
|
})
|
|
|
|
|
2020-12-07 07:32:36 -06:00
|
|
|
return res.type('json').status(HttpStatusCode.NO_CONTENT_204).end()
|
2018-04-25 09:15:39 -05:00
|
|
|
}
|
|
|
|
|
2019-03-19 04:35:15 -05:00
|
|
|
async function getVideoChannel (req: express.Request, res: express.Response) {
|
2018-04-25 09:15:39 -05:00
|
|
|
const videoChannelWithVideos = await VideoChannelModel.loadAndPopulateAccountAndVideos(res.locals.videoChannel.id)
|
|
|
|
|
2019-01-14 04:30:15 -06:00
|
|
|
if (videoChannelWithVideos.isOutdated()) {
|
|
|
|
JobQueue.Instance.createJob({ type: 'activitypub-refresher', payload: { type: 'actor', url: videoChannelWithVideos.Actor.url } })
|
|
|
|
}
|
|
|
|
|
2018-04-25 09:15:39 -05:00
|
|
|
return res.json(videoChannelWithVideos.toFormattedJSON())
|
|
|
|
}
|
|
|
|
|
2019-02-26 03:55:40 -06:00
|
|
|
async function listVideoChannelPlaylists (req: express.Request, res: express.Response) {
|
|
|
|
const serverActor = await getServerActor()
|
|
|
|
|
|
|
|
const resultList = await VideoPlaylistModel.listForApi({
|
|
|
|
followerActorId: serverActor.id,
|
|
|
|
start: req.query.start,
|
|
|
|
count: req.query.count,
|
|
|
|
sort: req.query.sort,
|
2019-03-05 03:58:44 -06:00
|
|
|
videoChannelId: res.locals.videoChannel.id,
|
|
|
|
type: req.query.playlistType
|
2019-02-26 03:55:40 -06:00
|
|
|
})
|
|
|
|
|
|
|
|
return res.json(getFormattedObjects(resultList.data, resultList.total))
|
|
|
|
}
|
|
|
|
|
2019-03-19 04:35:15 -05:00
|
|
|
async function listVideoChannelVideos (req: express.Request, res: express.Response) {
|
|
|
|
const videoChannelInstance = res.locals.videoChannel
|
2018-12-05 07:36:05 -06:00
|
|
|
const followerActorId = isUserAbleToSearchRemoteURI(res) ? null : undefined
|
2020-01-08 07:15:16 -06:00
|
|
|
const countVideos = getCountVideos(req)
|
2018-04-25 09:15:39 -05:00
|
|
|
|
2020-12-08 03:30:33 -06:00
|
|
|
const apiOptions = await Hooks.wrapObject({
|
2018-12-05 07:36:05 -06:00
|
|
|
followerActorId,
|
2018-04-25 09:15:39 -05:00
|
|
|
start: req.query.start,
|
|
|
|
count: req.query.count,
|
|
|
|
sort: req.query.sort,
|
2018-08-17 08:45:42 -05:00
|
|
|
includeLocalVideos: true,
|
2018-07-20 07:35:18 -05:00
|
|
|
categoryOneOf: req.query.categoryOneOf,
|
|
|
|
licenceOneOf: req.query.licenceOneOf,
|
|
|
|
languageOneOf: req.query.languageOneOf,
|
|
|
|
tagsOneOf: req.query.tagsOneOf,
|
|
|
|
tagsAllOf: req.query.tagsAllOf,
|
2018-10-10 04:46:50 -05:00
|
|
|
filter: req.query.filter,
|
2018-07-20 07:35:18 -05:00
|
|
|
nsfw: buildNSFWFilter(res, req.query.nsfw),
|
2018-04-25 09:15:39 -05:00
|
|
|
withFiles: false,
|
2018-10-10 04:46:50 -05:00
|
|
|
videoChannelId: videoChannelInstance.id,
|
2020-01-08 07:15:16 -06:00
|
|
|
user: res.locals.oauth ? res.locals.oauth.token.User : undefined,
|
|
|
|
countVideos
|
2020-12-08 03:30:33 -06:00
|
|
|
}, 'filter:api.video-channels.videos.list.params')
|
|
|
|
|
|
|
|
const resultList = await Hooks.wrapPromiseFun(
|
|
|
|
VideoModel.listForApi,
|
|
|
|
apiOptions,
|
|
|
|
'filter:api.video-channels.videos.list.result'
|
|
|
|
)
|
2018-04-25 09:15:39 -05:00
|
|
|
|
|
|
|
return res.json(getFormattedObjects(resultList.data, resultList.total))
|
|
|
|
}
|