2018-04-24 17:05:32 +02:00
|
|
|
import * as express from 'express'
|
2018-08-23 17:58:39 +02:00
|
|
|
import { getFormattedObjects, getServerActor } from '../../helpers/utils'
|
2018-04-24 17:05:32 +02:00
|
|
|
import {
|
|
|
|
asyncMiddleware,
|
2018-06-13 14:27:40 +02:00
|
|
|
asyncRetryTransactionMiddleware,
|
2018-08-14 15:28:30 +02:00
|
|
|
authenticate,
|
|
|
|
commonVideosFiltersValidator,
|
2018-04-25 16:15:39 +02:00
|
|
|
optionalAuthenticate,
|
2018-04-24 17:05:32 +02:00
|
|
|
paginationValidator,
|
|
|
|
setDefaultPagination,
|
|
|
|
setDefaultSort,
|
2018-04-25 16:15:39 +02:00
|
|
|
videoChannelsAddValidator,
|
|
|
|
videoChannelsRemoveValidator,
|
|
|
|
videoChannelsSortValidator,
|
|
|
|
videoChannelsUpdateValidator
|
2018-04-24 17:05:32 +02:00
|
|
|
} from '../../middlewares'
|
|
|
|
import { VideoChannelModel } from '../../models/video/video-channel'
|
2018-08-17 15:45:42 +02:00
|
|
|
import { videoChannelsNameWithHostValidator, videosSortValidator } from '../../middlewares/validators'
|
2018-04-25 16:15:39 +02:00
|
|
|
import { sendUpdateActor } from '../../lib/activitypub/send'
|
|
|
|
import { VideoChannelCreate, VideoChannelUpdate } from '../../../shared'
|
|
|
|
import { createVideoChannel } from '../../lib/video-channel'
|
2018-08-24 15:36:50 +02:00
|
|
|
import { buildNSFWFilter, createReqFiles, isUserAbleToSearchRemoteURI } from '../../helpers/express-utils'
|
2018-04-25 16:15:39 +02:00
|
|
|
import { setAsyncActorKeys } from '../../lib/activitypub'
|
|
|
|
import { AccountModel } from '../../models/account/account'
|
2018-06-29 11:29:23 +02:00
|
|
|
import { CONFIG, IMAGE_MIMETYPE_EXT, sequelizeTypescript } from '../../initializers'
|
2018-04-25 16:15:39 +02:00
|
|
|
import { logger } from '../../helpers/logger'
|
|
|
|
import { VideoModel } from '../../models/video/video'
|
2018-06-29 11:29:23 +02:00
|
|
|
import { updateAvatarValidator } from '../../middlewares/validators/avatar'
|
|
|
|
import { updateActorAvatarFile } from '../../lib/avatar'
|
2018-07-31 14:04:26 +02:00
|
|
|
import { auditLoggerFactory, VideoChannelAuditView } from '../../helpers/audit-logger'
|
2018-08-14 15:28:30 +02:00
|
|
|
import { resetSequelizeInstance } from '../../helpers/database-utils'
|
2018-06-29 11:29:23 +02:00
|
|
|
|
2018-07-31 14:04:26 +02:00
|
|
|
const auditLogger = auditLoggerFactory('channels')
|
2018-06-29 11:29:23 +02:00
|
|
|
const reqAvatarFile = createReqFiles([ 'avatarfile' ], IMAGE_MIMETYPE_EXT, { avatarfile: CONFIG.STORAGE.AVATARS_DIR })
|
2018-04-24 17:05:32 +02:00
|
|
|
|
|
|
|
const videoChannelRouter = express.Router()
|
|
|
|
|
|
|
|
videoChannelRouter.get('/',
|
|
|
|
paginationValidator,
|
|
|
|
videoChannelsSortValidator,
|
|
|
|
setDefaultSort,
|
|
|
|
setDefaultPagination,
|
|
|
|
asyncMiddleware(listVideoChannels)
|
|
|
|
)
|
|
|
|
|
2018-04-25 16:15:39 +02:00
|
|
|
videoChannelRouter.post('/',
|
|
|
|
authenticate,
|
|
|
|
videoChannelsAddValidator,
|
2018-06-13 14:27:40 +02:00
|
|
|
asyncRetryTransactionMiddleware(addVideoChannel)
|
2018-04-25 16:15:39 +02:00
|
|
|
)
|
|
|
|
|
2018-08-17 15:45:42 +02:00
|
|
|
videoChannelRouter.post('/:nameWithHost/avatar/pick',
|
2018-06-29 11:29:23 +02:00
|
|
|
authenticate,
|
|
|
|
reqAvatarFile,
|
|
|
|
// Check the rights
|
|
|
|
asyncMiddleware(videoChannelsUpdateValidator),
|
|
|
|
updateAvatarValidator,
|
|
|
|
asyncMiddleware(updateVideoChannelAvatar)
|
|
|
|
)
|
|
|
|
|
2018-08-17 15:45:42 +02:00
|
|
|
videoChannelRouter.put('/:nameWithHost',
|
2018-04-25 16:15:39 +02:00
|
|
|
authenticate,
|
|
|
|
asyncMiddleware(videoChannelsUpdateValidator),
|
2018-06-13 14:27:40 +02:00
|
|
|
asyncRetryTransactionMiddleware(updateVideoChannel)
|
2018-04-25 16:15:39 +02:00
|
|
|
)
|
|
|
|
|
2018-08-17 15:45:42 +02:00
|
|
|
videoChannelRouter.delete('/:nameWithHost',
|
2018-04-25 16:15:39 +02:00
|
|
|
authenticate,
|
|
|
|
asyncMiddleware(videoChannelsRemoveValidator),
|
2018-06-13 14:27:40 +02:00
|
|
|
asyncRetryTransactionMiddleware(removeVideoChannel)
|
2018-04-25 16:15:39 +02:00
|
|
|
)
|
|
|
|
|
2018-08-17 15:45:42 +02:00
|
|
|
videoChannelRouter.get('/:nameWithHost',
|
|
|
|
asyncMiddleware(videoChannelsNameWithHostValidator),
|
2018-04-25 16:15:39 +02:00
|
|
|
asyncMiddleware(getVideoChannel)
|
|
|
|
)
|
|
|
|
|
2018-08-17 15:45:42 +02:00
|
|
|
videoChannelRouter.get('/:nameWithHost/videos',
|
|
|
|
asyncMiddleware(videoChannelsNameWithHostValidator),
|
2018-04-25 16:15:39 +02:00
|
|
|
paginationValidator,
|
|
|
|
videosSortValidator,
|
|
|
|
setDefaultSort,
|
|
|
|
setDefaultPagination,
|
|
|
|
optionalAuthenticate,
|
2018-07-20 14:35:18 +02:00
|
|
|
commonVideosFiltersValidator,
|
2018-04-25 16:15:39 +02:00
|
|
|
asyncMiddleware(listVideoChannelVideos)
|
|
|
|
)
|
|
|
|
|
2018-04-24 17:05:32 +02:00
|
|
|
// ---------------------------------------------------------------------------
|
|
|
|
|
|
|
|
export {
|
|
|
|
videoChannelRouter
|
|
|
|
}
|
|
|
|
|
|
|
|
// ---------------------------------------------------------------------------
|
|
|
|
|
|
|
|
async function listVideoChannels (req: express.Request, res: express.Response, next: express.NextFunction) {
|
2018-08-23 17:58:39 +02:00
|
|
|
const serverActor = await getServerActor()
|
|
|
|
const resultList = await VideoChannelModel.listForApi(serverActor.id, req.query.start, req.query.count, req.query.sort)
|
2018-04-24 17:05:32 +02:00
|
|
|
|
|
|
|
return res.json(getFormattedObjects(resultList.data, resultList.total))
|
|
|
|
}
|
2018-04-25 16:15:39 +02:00
|
|
|
|
2018-06-29 11:29:23 +02:00
|
|
|
async function updateVideoChannelAvatar (req: express.Request, res: express.Response, next: express.NextFunction) {
|
|
|
|
const avatarPhysicalFile = req.files[ 'avatarfile' ][ 0 ]
|
2018-07-31 14:04:26 +02:00
|
|
|
const videoChannel = res.locals.videoChannel as VideoChannelModel
|
|
|
|
const oldVideoChannelAuditKeys = new VideoChannelAuditView(videoChannel.toFormattedJSON())
|
2018-06-29 11:29:23 +02:00
|
|
|
|
|
|
|
const avatar = await updateActorAvatarFile(avatarPhysicalFile, videoChannel.Actor, videoChannel)
|
|
|
|
|
2018-07-31 14:04:26 +02:00
|
|
|
auditLogger.update(
|
|
|
|
res.locals.oauth.token.User.Account.Actor.getIdentifier(),
|
|
|
|
new VideoChannelAuditView(videoChannel.toFormattedJSON()),
|
|
|
|
oldVideoChannelAuditKeys
|
|
|
|
)
|
|
|
|
|
2018-06-29 11:29:23 +02:00
|
|
|
return res
|
|
|
|
.json({
|
|
|
|
avatar: avatar.toFormattedJSON()
|
|
|
|
})
|
|
|
|
.end()
|
|
|
|
}
|
|
|
|
|
2018-04-25 16:15:39 +02:00
|
|
|
async function addVideoChannel (req: express.Request, res: express.Response) {
|
|
|
|
const videoChannelInfo: VideoChannelCreate = req.body
|
|
|
|
const account: AccountModel = res.locals.oauth.token.User.Account
|
|
|
|
|
|
|
|
const videoChannelCreated: VideoChannelModel = await sequelizeTypescript.transaction(async t => {
|
|
|
|
return createVideoChannel(videoChannelInfo, account, t)
|
|
|
|
})
|
|
|
|
|
|
|
|
setAsyncActorKeys(videoChannelCreated.Actor)
|
|
|
|
.catch(err => logger.error('Cannot set async actor keys for account %s.', videoChannelCreated.Actor.uuid, { err }))
|
|
|
|
|
2018-07-31 14:04:26 +02:00
|
|
|
auditLogger.create(
|
|
|
|
res.locals.oauth.token.User.Account.Actor.getIdentifier(),
|
|
|
|
new VideoChannelAuditView(videoChannelCreated.toFormattedJSON())
|
|
|
|
)
|
2018-04-25 16:15:39 +02:00
|
|
|
logger.info('Video channel with uuid %s created.', videoChannelCreated.Actor.uuid)
|
|
|
|
|
2018-06-13 14:27:40 +02:00
|
|
|
return res.json({
|
|
|
|
videoChannel: {
|
|
|
|
id: videoChannelCreated.id,
|
|
|
|
uuid: videoChannelCreated.Actor.uuid
|
|
|
|
}
|
|
|
|
}).end()
|
2018-04-25 16:15:39 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
async function updateVideoChannel (req: express.Request, res: express.Response) {
|
|
|
|
const videoChannelInstance = res.locals.videoChannel as VideoChannelModel
|
|
|
|
const videoChannelFieldsSave = videoChannelInstance.toJSON()
|
2018-07-31 14:04:26 +02:00
|
|
|
const oldVideoChannelAuditKeys = new VideoChannelAuditView(videoChannelInstance.toFormattedJSON())
|
2018-04-25 16:15:39 +02:00
|
|
|
const videoChannelInfoToUpdate = req.body as VideoChannelUpdate
|
|
|
|
|
|
|
|
try {
|
|
|
|
await sequelizeTypescript.transaction(async t => {
|
|
|
|
const sequelizeOptions = {
|
|
|
|
transaction: t
|
|
|
|
}
|
|
|
|
|
2018-04-26 16:11:38 +02:00
|
|
|
if (videoChannelInfoToUpdate.displayName !== undefined) videoChannelInstance.set('name', videoChannelInfoToUpdate.displayName)
|
2018-04-25 16:15:39 +02:00
|
|
|
if (videoChannelInfoToUpdate.description !== undefined) videoChannelInstance.set('description', videoChannelInfoToUpdate.description)
|
|
|
|
if (videoChannelInfoToUpdate.support !== undefined) videoChannelInstance.set('support', videoChannelInfoToUpdate.support)
|
|
|
|
|
|
|
|
const videoChannelInstanceUpdated = await videoChannelInstance.save(sequelizeOptions)
|
|
|
|
await sendUpdateActor(videoChannelInstanceUpdated, t)
|
|
|
|
|
2018-07-31 14:04:26 +02:00
|
|
|
auditLogger.update(
|
|
|
|
res.locals.oauth.token.User.Account.Actor.getIdentifier(),
|
|
|
|
new VideoChannelAuditView(videoChannelInstanceUpdated.toFormattedJSON()),
|
|
|
|
oldVideoChannelAuditKeys
|
|
|
|
)
|
|
|
|
logger.info('Video channel with name %s and uuid %s updated.', videoChannelInstance.name, videoChannelInstance.Actor.uuid)
|
|
|
|
})
|
2018-04-25 16:15:39 +02: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
|
|
|
|
}
|
|
|
|
|
|
|
|
return res.type('json').status(204).end()
|
|
|
|
}
|
|
|
|
|
|
|
|
async function removeVideoChannel (req: express.Request, res: express.Response) {
|
|
|
|
const videoChannelInstance: VideoChannelModel = res.locals.videoChannel
|
|
|
|
|
2018-06-13 14:27:40 +02:00
|
|
|
await sequelizeTypescript.transaction(async t => {
|
2018-04-25 16:15:39 +02:00
|
|
|
await videoChannelInstance.destroy({ transaction: t })
|
|
|
|
|
2018-07-31 14:04:26 +02:00
|
|
|
auditLogger.delete(
|
|
|
|
res.locals.oauth.token.User.Account.Actor.getIdentifier(),
|
|
|
|
new VideoChannelAuditView(videoChannelInstance.toFormattedJSON())
|
|
|
|
)
|
2018-04-25 16:15:39 +02:00
|
|
|
logger.info('Video channel with name %s and uuid %s deleted.', videoChannelInstance.name, videoChannelInstance.Actor.uuid)
|
|
|
|
})
|
|
|
|
|
2018-06-13 14:27:40 +02:00
|
|
|
return res.type('json').status(204).end()
|
2018-04-25 16:15:39 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
async function getVideoChannel (req: express.Request, res: express.Response, next: express.NextFunction) {
|
|
|
|
const videoChannelWithVideos = await VideoChannelModel.loadAndPopulateAccountAndVideos(res.locals.videoChannel.id)
|
|
|
|
|
|
|
|
return res.json(videoChannelWithVideos.toFormattedJSON())
|
|
|
|
}
|
|
|
|
|
|
|
|
async function listVideoChannelVideos (req: express.Request, res: express.Response, next: express.NextFunction) {
|
|
|
|
const videoChannelInstance: VideoChannelModel = res.locals.videoChannel
|
2018-08-24 15:36:50 +02:00
|
|
|
const actorId = isUserAbleToSearchRemoteURI(res) ? null : undefined
|
2018-04-25 16:15:39 +02:00
|
|
|
|
|
|
|
const resultList = await VideoModel.listForApi({
|
2018-08-24 15:36:50 +02:00
|
|
|
actorId,
|
2018-04-25 16:15:39 +02:00
|
|
|
start: req.query.start,
|
|
|
|
count: req.query.count,
|
|
|
|
sort: req.query.sort,
|
2018-08-17 15:45:42 +02:00
|
|
|
includeLocalVideos: true,
|
2018-07-20 14:35:18 +02:00
|
|
|
categoryOneOf: req.query.categoryOneOf,
|
|
|
|
licenceOneOf: req.query.licenceOneOf,
|
|
|
|
languageOneOf: req.query.languageOneOf,
|
|
|
|
tagsOneOf: req.query.tagsOneOf,
|
|
|
|
tagsAllOf: req.query.tagsAllOf,
|
|
|
|
nsfw: buildNSFWFilter(res, req.query.nsfw),
|
2018-04-25 16:15:39 +02:00
|
|
|
withFiles: false,
|
|
|
|
videoChannelId: videoChannelInstance.id
|
|
|
|
})
|
|
|
|
|
|
|
|
return res.json(getFormattedObjects(resultList.data, resultList.total))
|
|
|
|
}
|