]> git.immae.eu Git - github/Chocobozzz/PeerTube.git/blame - server/controllers/api/videos/index.ts
Avoid concurrency issue on transcoding
[github/Chocobozzz/PeerTube.git] / server / controllers / api / videos / index.ts
CommitLineData
41fb13c3 1import express from 'express'
d6886027 2import { pickCommonVideoQuery } from '@server/helpers/query'
304a84d5 3import { doJSONRequest } from '@server/helpers/requests'
1c627fd8 4import { openapiOperationDoc } from '@server/middlewares/doc'
8054669f 5import { getServerActor } from '@server/models/application/application'
2760b454 6import { guessAdditionalAttributesFromQuery } from '@server/models/video/formatter/video-format-utils'
304a84d5 7import { MVideoAccountLight } from '@server/types/models'
c0e8b12e 8import { HttpStatusCode } from '../../../../shared/models'
8054669f 9import { auditLoggerFactory, getAuditIdFromRes, VideoAuditView } from '../../../helpers/audit-logger'
c158a5fa
C
10import { buildNSFWFilter, getCountVideos } from '../../../helpers/express-utils'
11import { logger } from '../../../helpers/logger'
8dc8a34e 12import { getFormattedObjects } from '../../../helpers/utils'
304a84d5 13import { REMOTE_SCHEME, VIDEO_CATEGORIES, VIDEO_LANGUAGES, VIDEO_LICENCES, VIDEO_PRIVACIES } from '../../../initializers/constants'
8054669f 14import { sequelizeTypescript } from '../../../initializers/database'
94a5ff8a 15import { JobQueue } from '../../../lib/job-queue'
8054669f 16import { Hooks } from '../../../lib/plugins/hooks'
65fcc311 17import {
ac81d1a0 18 asyncMiddleware,
90d4bb81 19 asyncRetryTransactionMiddleware,
ac81d1a0 20 authenticate,
8d427346 21 checkVideoFollowConstraints,
d525fc39 22 commonVideosFiltersValidator,
0883b324 23 optionalAuthenticate,
ac81d1a0
C
24 paginationValidator,
25 setDefaultPagination,
8054669f 26 setDefaultVideosSort,
09209296 27 videosCustomGetValidator,
ac81d1a0 28 videosGetValidator,
2e401e85 29 videoSourceGetValidator,
ac81d1a0 30 videosRemoveValidator,
c158a5fa 31 videosSortValidator
65fcc311 32} from '../../../middlewares'
3fd3ab2d 33import { VideoModel } from '../../../models/video/video'
65fcc311 34import { blacklistRouter } from './blacklist'
40e87e9e 35import { videoCaptionsRouter } from './captions'
8054669f 36import { videoCommentRouter } from './comment'
b46cf4b9 37import { filesRouter } from './files'
fbad87b0 38import { videoImportsRouter } from './import'
c6c0fa6c 39import { liveRouter } from './live'
8054669f
C
40import { ownershipVideoRouter } from './ownership'
41import { rateVideoRouter } from './rate'
b2111066
C
42import { statsRouter } from './stats'
43import { studioRouter } from './studio'
ad5db104 44import { transcodingRouter } from './transcoding'
c158a5fa
C
45import { updateRouter } from './update'
46import { uploadRouter } from './upload'
b2111066 47import { viewRouter } from './view'
65fcc311 48
80e36cd9 49const auditLogger = auditLoggerFactory('videos')
65fcc311 50const videosRouter = express.Router()
8c308c2b 51
65fcc311 52videosRouter.use('/', blacklistRouter)
b2111066 53videosRouter.use('/', statsRouter)
65fcc311 54videosRouter.use('/', rateVideoRouter)
bf1f6508 55videosRouter.use('/', videoCommentRouter)
92e66e04 56videosRouter.use('/', studioRouter)
40e87e9e 57videosRouter.use('/', videoCaptionsRouter)
fbad87b0 58videosRouter.use('/', videoImportsRouter)
74d63469 59videosRouter.use('/', ownershipVideoRouter)
b2111066 60videosRouter.use('/', viewRouter)
c6c0fa6c 61videosRouter.use('/', liveRouter)
c158a5fa
C
62videosRouter.use('/', uploadRouter)
63videosRouter.use('/', updateRouter)
b46cf4b9 64videosRouter.use('/', filesRouter)
ad5db104 65videosRouter.use('/', transcodingRouter)
d33242b0 66
c756bae0
RK
67videosRouter.get('/categories',
68 openapiOperationDoc({ operationId: 'getCategories' }),
69 listVideoCategories
70)
71videosRouter.get('/licences',
72 openapiOperationDoc({ operationId: 'getLicences' }),
73 listVideoLicences
74)
75videosRouter.get('/languages',
76 openapiOperationDoc({ operationId: 'getLanguages' }),
77 listVideoLanguages
78)
79videosRouter.get('/privacies',
80 openapiOperationDoc({ operationId: 'getPrivacies' }),
81 listVideoPrivacies
82)
6e07c3de 83
65fcc311 84videosRouter.get('/',
c756bae0 85 openapiOperationDoc({ operationId: 'getVideos' }),
65fcc311
C
86 paginationValidator,
87 videosSortValidator,
8054669f 88 setDefaultVideosSort,
f05a1c30 89 setDefaultPagination,
0883b324 90 optionalAuthenticate,
d525fc39 91 commonVideosFiltersValidator,
eb080476 92 asyncMiddleware(listVideos)
fbf1134e 93)
f6d6e7f8 94
9567011b 95videosRouter.get('/:id/description',
c756bae0 96 openapiOperationDoc({ operationId: 'getVideoDesc' }),
a2431b7d 97 asyncMiddleware(videosGetValidator),
9567011b
C
98 asyncMiddleware(getVideoDescription)
99)
2e401e85 100
101videosRouter.get('/:id/source',
102 openapiOperationDoc({ operationId: 'getVideoSource' }),
103 authenticate,
104 asyncMiddleware(videoSourceGetValidator),
105 getVideoSource
106)
107
65fcc311 108videosRouter.get('/:id',
1c627fd8 109 openapiOperationDoc({ operationId: 'getVideo' }),
6e46de09 110 optionalAuthenticate,
ca4b4b2e 111 asyncMiddleware(videosCustomGetValidator('for-api')),
8d427346 112 asyncMiddleware(checkVideoFollowConstraints),
0260dc8a 113 asyncMiddleware(getVideo)
fbf1134e 114)
198b205c 115
65fcc311 116videosRouter.delete('/:id',
1c627fd8 117 openapiOperationDoc({ operationId: 'delVideo' }),
65fcc311 118 authenticate,
a2431b7d 119 asyncMiddleware(videosRemoveValidator),
90d4bb81 120 asyncRetryTransactionMiddleware(removeVideo)
fbf1134e 121)
198b205c 122
9f10b292 123// ---------------------------------------------------------------------------
c45f7f84 124
65fcc311
C
125export {
126 videosRouter
127}
c45f7f84 128
9f10b292 129// ---------------------------------------------------------------------------
c45f7f84 130
f6d6e7f8 131function listVideoCategories (_req: express.Request, res: express.Response) {
65fcc311 132 res.json(VIDEO_CATEGORIES)
6e07c3de
C
133}
134
f6d6e7f8 135function listVideoLicences (_req: express.Request, res: express.Response) {
65fcc311 136 res.json(VIDEO_LICENCES)
6f0c39e2
C
137}
138
f6d6e7f8 139function listVideoLanguages (_req: express.Request, res: express.Response) {
65fcc311 140 res.json(VIDEO_LANGUAGES)
3092476e
C
141}
142
f6d6e7f8 143function listVideoPrivacies (_req: express.Request, res: express.Response) {
fd45e8f4
C
144 res.json(VIDEO_PRIVACIES)
145}
146
0260dc8a
C
147async function getVideo (_req: express.Request, res: express.Response) {
148 const videoId = res.locals.videoAPI.id
149 const userId = res.locals.oauth?.token.User.id
150
151 const video = await Hooks.wrapObject(res.locals.videoAPI, 'filter:api.video.get.result', { id: videoId, userId })
1f3e9fec 152
09209296 153 if (video.isOutdated()) {
bd911b54 154 JobQueue.Instance.createJobAsync({ type: 'activitypub-refresher', payload: { type: 'video', url: video.url } })
04b8c3fb
C
155 }
156
09209296 157 return res.json(video.toFormattedDetailsJSON())
1f3e9fec
C
158}
159
9567011b 160async function getVideoDescription (req: express.Request, res: express.Response) {
453e83ea 161 const videoInstance = res.locals.videoAll
9567011b 162
c158a5fa
C
163 const description = videoInstance.isOwned()
164 ? videoInstance.description
165 : await fetchRemoteVideoDescription(videoInstance)
9567011b
C
166
167 return res.json({ description })
168}
169
2e401e85 170function getVideoSource (req: express.Request, res: express.Response) {
171 return res.json(res.locals.videoSource.toFormattedJSON())
172}
173
04b8c3fb 174async function listVideos (req: express.Request, res: express.Response) {
2760b454
C
175 const serverActor = await getServerActor()
176
d6886027 177 const query = pickCommonVideoQuery(req.query)
fe987656
C
178 const countVideos = getCountVideos(req)
179
b4055e1c 180 const apiOptions = await Hooks.wrapObject({
d6886027
C
181 ...query,
182
2760b454
C
183 displayOnlyForFollower: {
184 actorId: serverActor.id,
185 orLocalVideos: true
186 },
1fd61899 187 nsfw: buildNSFWFilter(res, query.nsfw),
fe987656
C
188 user: res.locals.oauth ? res.locals.oauth.token.User : undefined,
189 countVideos
b4055e1c
C
190 }, 'filter:api.videos.list.params')
191
89cd1275
C
192 const resultList = await Hooks.wrapPromiseFun(
193 VideoModel.listForApi,
194 apiOptions,
b4055e1c
C
195 'filter:api.videos.list.result'
196 )
eb080476 197
2760b454 198 return res.json(getFormattedObjects(resultList.data, resultList.total, guessAdditionalAttributesFromQuery(query)))
9f10b292 199}
c45f7f84 200
7226e90f 201async function removeVideo (req: express.Request, res: express.Response) {
453e83ea 202 const videoInstance = res.locals.videoAll
91f6f169 203
3fd3ab2d 204 await sequelizeTypescript.transaction(async t => {
eb080476 205 await videoInstance.destroy({ transaction: t })
91f6f169 206 })
eb080476 207
993cef4b 208 auditLogger.delete(getAuditIdFromRes(res), new VideoAuditView(videoInstance.toFormattedDetailsJSON()))
eb080476 209 logger.info('Video with name %s and uuid %s deleted.', videoInstance.name, videoInstance.uuid)
90d4bb81 210
7226e90f 211 Hooks.runAction('action:api.video.deleted', { video: videoInstance, req, res })
b4055e1c 212
2d53be02
RK
213 return res.type('json')
214 .status(HttpStatusCode.NO_CONTENT_204)
215 .end()
9f10b292 216}
304a84d5
C
217
218// ---------------------------------------------------------------------------
219
220// FIXME: Should not exist, we rely on specific API
221async function fetchRemoteVideoDescription (video: MVideoAccountLight) {
222 const host = video.VideoChannel.Account.Actor.Server.host
223 const path = video.getDescriptionAPIPath()
224 const url = REMOTE_SCHEME.HTTP + '://' + host + path
225
226 const { body } = await doJSONRequest<any>(url)
227 return body.description || ''
228}