]> git.immae.eu Git - github/Chocobozzz/PeerTube.git/blame_incremental - server/middlewares/validators/videos/video-channels.ts
Reorganize plugin models
[github/Chocobozzz/PeerTube.git] / server / middlewares / validators / videos / video-channels.ts
... / ...
CommitLineData
1import * as express from 'express'
2import { body, param, query } from 'express-validator'
3import { VIDEO_CHANNELS } from '@server/initializers/constants'
4import { MChannelAccountDefault, MUser } from '@server/types/models'
5import { UserRight } from '../../../../shared'
6import { HttpStatusCode } from '../../../../shared/core-utils/miscs/http-error-codes'
7import { isActorPreferredUsernameValid } from '../../../helpers/custom-validators/activitypub/actor'
8import { isBooleanValid, toBooleanOrNull } from '../../../helpers/custom-validators/misc'
9import {
10 isVideoChannelDescriptionValid,
11 isVideoChannelNameValid,
12 isVideoChannelSupportValid
13} from '../../../helpers/custom-validators/video-channels'
14import { logger } from '../../../helpers/logger'
15import { doesLocalVideoChannelNameExist, doesVideoChannelNameWithHostExist } from '../../../helpers/middlewares'
16import { ActorModel } from '../../../models/actor/actor'
17import { VideoChannelModel } from '../../../models/video/video-channel'
18import { areValidationErrors } from '../utils'
19
20const videoChannelsAddValidator = [
21 body('name').custom(isActorPreferredUsernameValid).withMessage('Should have a valid channel name'),
22 body('displayName').custom(isVideoChannelNameValid).withMessage('Should have a valid display name'),
23 body('description').optional().custom(isVideoChannelDescriptionValid).withMessage('Should have a valid description'),
24 body('support').optional().custom(isVideoChannelSupportValid).withMessage('Should have a valid support text'),
25
26 async (req: express.Request, res: express.Response, next: express.NextFunction) => {
27 logger.debug('Checking videoChannelsAdd parameters', { parameters: req.body })
28
29 if (areValidationErrors(req, res)) return
30
31 const actor = await ActorModel.loadLocalByName(req.body.name)
32 if (actor) {
33 res.status(HttpStatusCode.CONFLICT_409)
34 .send({ error: 'Another actor (account/channel) with this name on this instance already exists or has already existed.' })
35 .end()
36 return false
37 }
38
39 const count = await VideoChannelModel.countByAccount(res.locals.oauth.token.User.Account.id)
40 if (count >= VIDEO_CHANNELS.MAX_PER_USER) {
41 res.status(HttpStatusCode.BAD_REQUEST_400)
42 .send({ error: `You cannot create more than ${VIDEO_CHANNELS.MAX_PER_USER} channels` })
43 .end()
44 return false
45 }
46
47 return next()
48 }
49]
50
51const videoChannelsUpdateValidator = [
52 param('nameWithHost').exists().withMessage('Should have an video channel name with host'),
53 body('displayName')
54 .optional()
55 .custom(isVideoChannelNameValid).withMessage('Should have a valid display name'),
56 body('description')
57 .optional()
58 .custom(isVideoChannelDescriptionValid).withMessage('Should have a valid description'),
59 body('support')
60 .optional()
61 .custom(isVideoChannelSupportValid).withMessage('Should have a valid support text'),
62 body('bulkVideosSupportUpdate')
63 .optional()
64 .custom(isBooleanValid).withMessage('Should have a valid bulkVideosSupportUpdate boolean field'),
65
66 async (req: express.Request, res: express.Response, next: express.NextFunction) => {
67 logger.debug('Checking videoChannelsUpdate parameters', { parameters: req.body })
68
69 if (areValidationErrors(req, res)) return
70 if (!await doesVideoChannelNameWithHostExist(req.params.nameWithHost, res)) return
71
72 // We need to make additional checks
73 if (res.locals.videoChannel.Actor.isOwned() === false) {
74 return res.status(HttpStatusCode.FORBIDDEN_403)
75 .json({ error: 'Cannot update video channel of another server' })
76 }
77
78 if (res.locals.videoChannel.Account.userId !== res.locals.oauth.token.User.id) {
79 return res.status(HttpStatusCode.FORBIDDEN_403)
80 .json({ error: 'Cannot update video channel of another user' })
81 }
82
83 return next()
84 }
85]
86
87const videoChannelsRemoveValidator = [
88 param('nameWithHost').exists().withMessage('Should have an video channel name with host'),
89
90 async (req: express.Request, res: express.Response, next: express.NextFunction) => {
91 logger.debug('Checking videoChannelsRemove parameters', { parameters: req.params })
92
93 if (areValidationErrors(req, res)) return
94 if (!await doesVideoChannelNameWithHostExist(req.params.nameWithHost, res)) return
95
96 if (!checkUserCanDeleteVideoChannel(res.locals.oauth.token.User, res.locals.videoChannel, res)) return
97 if (!await checkVideoChannelIsNotTheLastOne(res)) return
98
99 return next()
100 }
101]
102
103const videoChannelsNameWithHostValidator = [
104 param('nameWithHost').exists().withMessage('Should have an video channel name with host'),
105
106 async (req: express.Request, res: express.Response, next: express.NextFunction) => {
107 logger.debug('Checking videoChannelsNameWithHostValidator parameters', { parameters: req.params })
108
109 if (areValidationErrors(req, res)) return
110
111 if (!await doesVideoChannelNameWithHostExist(req.params.nameWithHost, res)) return
112
113 return next()
114 }
115]
116
117const localVideoChannelValidator = [
118 param('name').custom(isVideoChannelNameValid).withMessage('Should have a valid video channel name'),
119
120 async (req: express.Request, res: express.Response, next: express.NextFunction) => {
121 logger.debug('Checking localVideoChannelValidator parameters', { parameters: req.params })
122
123 if (areValidationErrors(req, res)) return
124 if (!await doesLocalVideoChannelNameExist(req.params.name, res)) return
125
126 return next()
127 }
128]
129
130const videoChannelStatsValidator = [
131 query('withStats')
132 .optional()
133 .customSanitizer(toBooleanOrNull)
134 .custom(isBooleanValid).withMessage('Should have a valid stats flag'),
135
136 (req: express.Request, res: express.Response, next: express.NextFunction) => {
137 if (areValidationErrors(req, res)) return
138 return next()
139 }
140]
141
142// ---------------------------------------------------------------------------
143
144export {
145 videoChannelsAddValidator,
146 videoChannelsUpdateValidator,
147 videoChannelsRemoveValidator,
148 videoChannelsNameWithHostValidator,
149 localVideoChannelValidator,
150 videoChannelStatsValidator
151}
152
153// ---------------------------------------------------------------------------
154
155function checkUserCanDeleteVideoChannel (user: MUser, videoChannel: MChannelAccountDefault, res: express.Response) {
156 if (videoChannel.Actor.isOwned() === false) {
157 res.status(HttpStatusCode.FORBIDDEN_403)
158 .json({ error: 'Cannot remove video channel of another server.' })
159 .end()
160
161 return false
162 }
163
164 // Check if the user can delete the video channel
165 // The user can delete it if s/he is an admin
166 // Or if s/he is the video channel's account
167 if (user.hasRight(UserRight.REMOVE_ANY_VIDEO_CHANNEL) === false && videoChannel.Account.userId !== user.id) {
168 res.status(HttpStatusCode.FORBIDDEN_403)
169 .json({ error: 'Cannot remove video channel of another user' })
170 .end()
171
172 return false
173 }
174
175 return true
176}
177
178async function checkVideoChannelIsNotTheLastOne (res: express.Response) {
179 const count = await VideoChannelModel.countByAccount(res.locals.oauth.token.User.Account.id)
180
181 if (count <= 1) {
182 res.status(HttpStatusCode.CONFLICT_409)
183 .json({ error: 'Cannot remove the last channel of this user' })
184 .end()
185
186 return false
187 }
188
189 return true
190}