]>
Commit | Line | Data |
---|---|---|
41fb13c3 | 1 | import express from 'express' |
f443a746 | 2 | import { exists } from '@server/helpers/custom-validators/misc' |
c6c0fa6c | 3 | import { createReqFiles } from '@server/helpers/express-utils' |
c6c0fa6c | 4 | import { ASSETS_PATH, MIMETYPES } from '@server/initializers/constants' |
de94ac86 | 5 | import { getLocalVideoActivityPubUrl } from '@server/lib/activitypub/url' |
af4ae64f | 6 | import { federateVideoIfNeeded } from '@server/lib/activitypub/videos' |
3cabf353 | 7 | import { Hooks } from '@server/lib/plugins/hooks' |
1ef65f4c | 8 | import { buildLocalVideoFromReq, buildVideoThumbnailsFromReq, setVideoTags } from '@server/lib/video' |
b5b68755 | 9 | import { videoLiveAddValidator, videoLiveGetValidator, videoLiveUpdateValidator } from '@server/middlewares/validators/videos/video-live' |
c6c0fa6c C |
10 | import { VideoLiveModel } from '@server/models/video/video-live' |
11 | import { MVideoDetails, MVideoFullLight } from '@server/types/models' | |
0628157f | 12 | import { buildUUID, uuidToShort } from '@shared/extra-utils' |
961cbe42 | 13 | import { HttpStatusCode, LiveVideoCreate, LiveVideoLatencyMode, LiveVideoUpdate, UserRight, VideoState } from '@shared/models' |
c6c0fa6c C |
14 | import { logger } from '../../../helpers/logger' |
15 | import { sequelizeTypescript } from '../../../initializers/database' | |
91f8f8db | 16 | import { updateVideoMiniatureFromExisting } from '../../../lib/thumbnail' |
961cbe42 | 17 | import { asyncMiddleware, asyncRetryTransactionMiddleware, authenticate, optionalAuthenticate } from '../../../middlewares' |
c6c0fa6c | 18 | import { VideoModel } from '../../../models/video/video' |
c6c0fa6c C |
19 | |
20 | const liveRouter = express.Router() | |
21 | ||
d3d3deaa | 22 | const reqVideoFileLive = createReqFiles([ 'thumbnailfile', 'previewfile' ], MIMETYPES.IMAGE.MIMETYPE_EXT) |
c6c0fa6c C |
23 | |
24 | liveRouter.post('/live', | |
25 | authenticate, | |
26 | reqVideoFileLive, | |
27 | asyncMiddleware(videoLiveAddValidator), | |
28 | asyncRetryTransactionMiddleware(addLiveVideo) | |
29 | ) | |
30 | ||
31 | liveRouter.get('/live/:videoId', | |
961cbe42 | 32 | optionalAuthenticate, |
c6c0fa6c | 33 | asyncMiddleware(videoLiveGetValidator), |
98ab5dc8 | 34 | getLiveVideo |
b5b68755 C |
35 | ) |
36 | ||
37 | liveRouter.put('/live/:videoId', | |
38 | authenticate, | |
39 | asyncMiddleware(videoLiveGetValidator), | |
40 | videoLiveUpdateValidator, | |
41 | asyncRetryTransactionMiddleware(updateLiveVideo) | |
c6c0fa6c C |
42 | ) |
43 | ||
44 | // --------------------------------------------------------------------------- | |
45 | ||
46 | export { | |
47 | liveRouter | |
48 | } | |
49 | ||
50 | // --------------------------------------------------------------------------- | |
51 | ||
98ab5dc8 | 52 | function getLiveVideo (req: express.Request, res: express.Response) { |
c6c0fa6c C |
53 | const videoLive = res.locals.videoLive |
54 | ||
961cbe42 C |
55 | return res.json(videoLive.toFormattedJSON(canSeePrivateLiveInformation(res))) |
56 | } | |
57 | ||
58 | function canSeePrivateLiveInformation (res: express.Response) { | |
59 | const user = res.locals.oauth?.token.User | |
60 | if (!user) return false | |
61 | ||
62 | if (user.hasRight(UserRight.GET_ANY_LIVE)) return true | |
63 | ||
64 | const video = res.locals.videoAll | |
65 | return video.VideoChannel.Account.userId === user.id | |
c6c0fa6c C |
66 | } |
67 | ||
b5b68755 C |
68 | async function updateLiveVideo (req: express.Request, res: express.Response) { |
69 | const body: LiveVideoUpdate = req.body | |
70 | ||
af4ae64f | 71 | const video = res.locals.videoAll |
b5b68755 | 72 | const videoLive = res.locals.videoLive |
bb4ba6d9 | 73 | |
f443a746 C |
74 | if (exists(body.saveReplay)) videoLive.saveReplay = body.saveReplay |
75 | if (exists(body.permanentLive)) videoLive.permanentLive = body.permanentLive | |
76 | if (exists(body.latencyMode)) videoLive.latencyMode = body.latencyMode | |
b5b68755 | 77 | |
af4ae64f C |
78 | video.VideoLive = await videoLive.save() |
79 | ||
80 | await federateVideoIfNeeded(video, false) | |
b5b68755 | 81 | |
76148b27 | 82 | return res.status(HttpStatusCode.NO_CONTENT_204).end() |
b5b68755 C |
83 | } |
84 | ||
c6c0fa6c | 85 | async function addLiveVideo (req: express.Request, res: express.Response) { |
b5b68755 | 86 | const videoInfo: LiveVideoCreate = req.body |
c6c0fa6c C |
87 | |
88 | // Prepare data so we don't block the transaction | |
d17d7430 C |
89 | let videoData = buildLocalVideoFromReq(videoInfo, res.locals.videoChannel.id) |
90 | videoData = await Hooks.wrapObject(videoData, 'filter:api.video.live.video-attribute.result') | |
91 | ||
c6c0fa6c | 92 | videoData.isLive = true |
1ef65f4c C |
93 | videoData.state = VideoState.WAITING_FOR_LIVE |
94 | videoData.duration = 0 | |
c6c0fa6c C |
95 | |
96 | const video = new VideoModel(videoData) as MVideoDetails | |
de94ac86 | 97 | video.url = getLocalVideoActivityPubUrl(video) // We use the UUID, so set the URL after building the object |
c6c0fa6c | 98 | |
1ef65f4c | 99 | const videoLive = new VideoLiveModel() |
b5b68755 | 100 | videoLive.saveReplay = videoInfo.saveReplay || false |
bb4ba6d9 | 101 | videoLive.permanentLive = videoInfo.permanentLive || false |
f443a746 | 102 | videoLive.latencyMode = videoInfo.latencyMode || LiveVideoLatencyMode.DEFAULT |
d4a8e7a6 | 103 | videoLive.streamKey = buildUUID() |
c6c0fa6c | 104 | |
1ef65f4c C |
105 | const [ thumbnailModel, previewModel ] = await buildVideoThumbnailsFromReq({ |
106 | video, | |
107 | files: req.files, | |
108 | fallback: type => { | |
91f8f8db | 109 | return updateVideoMiniatureFromExisting({ |
b5b68755 C |
110 | inputPath: ASSETS_PATH.DEFAULT_LIVE_BACKGROUND, |
111 | video, | |
112 | type, | |
113 | automaticallyGenerated: true, | |
114 | keepOriginal: true | |
115 | }) | |
1ef65f4c C |
116 | } |
117 | }) | |
c6c0fa6c C |
118 | |
119 | const { videoCreated } = await sequelizeTypescript.transaction(async t => { | |
120 | const sequelizeOptions = { transaction: t } | |
121 | ||
122 | const videoCreated = await video.save(sequelizeOptions) as MVideoFullLight | |
123 | ||
124 | if (thumbnailModel) await videoCreated.addAndSaveThumbnail(thumbnailModel, t) | |
125 | if (previewModel) await videoCreated.addAndSaveThumbnail(previewModel, t) | |
126 | ||
127 | // Do not forget to add video channel information to the created video | |
128 | videoCreated.VideoChannel = res.locals.videoChannel | |
129 | ||
130 | videoLive.videoId = videoCreated.id | |
af4ae64f | 131 | videoCreated.VideoLive = await videoLive.save(sequelizeOptions) |
c6c0fa6c | 132 | |
1ef65f4c | 133 | await setVideoTags({ video, tags: videoInfo.tags, transaction: t }) |
c6c0fa6c | 134 | |
af4ae64f C |
135 | await federateVideoIfNeeded(videoCreated, true, t) |
136 | ||
c6c0fa6c C |
137 | logger.info('Video live %s with uuid %s created.', videoInfo.name, videoCreated.uuid) |
138 | ||
139 | return { videoCreated } | |
140 | }) | |
141 | ||
7226e90f | 142 | Hooks.runAction('action:api.live-video.created', { video: videoCreated, req, res }) |
3cabf353 | 143 | |
c6c0fa6c C |
144 | return res.json({ |
145 | video: { | |
146 | id: videoCreated.id, | |
d4a8e7a6 | 147 | shortUUID: uuidToShort(videoCreated.uuid), |
c6c0fa6c C |
148 | uuid: videoCreated.uuid |
149 | } | |
150 | }) | |
151 | } |