]>
Commit | Line | Data |
---|---|---|
9f10b292 C |
1 | 'use strict' |
2 | ||
f0f5567b | 3 | const express = require('express') |
558d7c23 | 4 | const fs = require('fs') |
f0f5567b | 5 | const multer = require('multer') |
558d7c23 | 6 | const path = require('path') |
1a42c9e2 | 7 | const waterfall = require('async/waterfall') |
f0f5567b | 8 | |
f253b1c1 | 9 | const constants = require('../../initializers/constants') |
feb4bdfd | 10 | const db = require('../../initializers/database') |
f253b1c1 C |
11 | const logger = require('../../helpers/logger') |
12 | const friends = require('../../lib/friends') | |
13 | const middlewares = require('../../middlewares') | |
55fa55a9 | 14 | const admin = middlewares.admin |
69b0a27c | 15 | const oAuth = middlewares.oauth |
fbf1134e | 16 | const pagination = middlewares.pagination |
fc51fde0 C |
17 | const validators = middlewares.validators |
18 | const validatorsPagination = validators.pagination | |
19 | const validatorsSort = validators.sort | |
20 | const validatorsVideos = validators.videos | |
46246b5f | 21 | const search = middlewares.search |
a877d5ac | 22 | const sort = middlewares.sort |
f253b1c1 | 23 | const utils = require('../../helpers/utils') |
f0f5567b C |
24 | |
25 | const router = express.Router() | |
9f10b292 C |
26 | |
27 | // multer configuration | |
f0f5567b | 28 | const storage = multer.diskStorage({ |
9f10b292 | 29 | destination: function (req, file, cb) { |
b3d92510 | 30 | cb(null, constants.CONFIG.STORAGE.VIDEOS_DIR) |
9f10b292 C |
31 | }, |
32 | ||
33 | filename: function (req, file, cb) { | |
f0f5567b | 34 | let extension = '' |
9f10b292 C |
35 | if (file.mimetype === 'video/webm') extension = 'webm' |
36 | else if (file.mimetype === 'video/mp4') extension = 'mp4' | |
37 | else if (file.mimetype === 'video/ogg') extension = 'ogv' | |
bc503c2a C |
38 | utils.generateRandomString(16, function (err, randomString) { |
39 | const fieldname = err ? undefined : randomString | |
9f10b292 C |
40 | cb(null, fieldname + '.' + extension) |
41 | }) | |
42 | } | |
43 | }) | |
44 | ||
8c9c1942 | 45 | const reqFiles = multer({ storage: storage }).fields([{ name: 'videofile', maxCount: 1 }]) |
8c308c2b | 46 | |
55fa55a9 C |
47 | router.get('/abuse', |
48 | oAuth.authenticate, | |
49 | admin.ensureIsAdmin, | |
50 | validatorsPagination.pagination, | |
51 | validatorsSort.videoAbusesSort, | |
52 | sort.setVideoAbusesSort, | |
53 | pagination.setPagination, | |
54 | listVideoAbuses | |
55 | ) | |
56 | router.post('/:id/abuse', | |
57 | oAuth.authenticate, | |
58 | validatorsVideos.videoAbuseReport, | |
bf4ff8fe | 59 | reportVideoAbuseRetryWrapper |
55fa55a9 C |
60 | ) |
61 | ||
fbf1134e | 62 | router.get('/', |
fc51fde0 C |
63 | validatorsPagination.pagination, |
64 | validatorsSort.videosSort, | |
a877d5ac | 65 | sort.setVideosSort, |
fbf1134e C |
66 | pagination.setPagination, |
67 | listVideos | |
68 | ) | |
7b1f49de C |
69 | router.put('/:id', |
70 | oAuth.authenticate, | |
71 | reqFiles, | |
72 | validatorsVideos.videosUpdate, | |
ed04d94f | 73 | updateVideoRetryWrapper |
7b1f49de | 74 | ) |
fbf1134e | 75 | router.post('/', |
69b0a27c | 76 | oAuth.authenticate, |
fbf1134e | 77 | reqFiles, |
fc51fde0 | 78 | validatorsVideos.videosAdd, |
ed04d94f | 79 | addVideoRetryWrapper |
fbf1134e C |
80 | ) |
81 | router.get('/:id', | |
fc51fde0 | 82 | validatorsVideos.videosGet, |
68ce3ae0 | 83 | getVideo |
fbf1134e C |
84 | ) |
85 | router.delete('/:id', | |
69b0a27c | 86 | oAuth.authenticate, |
fc51fde0 | 87 | validatorsVideos.videosRemove, |
fbf1134e C |
88 | removeVideo |
89 | ) | |
46246b5f | 90 | router.get('/search/:value', |
fc51fde0 C |
91 | validatorsVideos.videosSearch, |
92 | validatorsPagination.pagination, | |
93 | validatorsSort.videosSort, | |
a877d5ac | 94 | sort.setVideosSort, |
fbf1134e | 95 | pagination.setPagination, |
46246b5f | 96 | search.setVideosSearch, |
fbf1134e C |
97 | searchVideos |
98 | ) | |
8c308c2b | 99 | |
9f10b292 | 100 | // --------------------------------------------------------------------------- |
c45f7f84 | 101 | |
9f10b292 | 102 | module.exports = router |
c45f7f84 | 103 | |
9f10b292 | 104 | // --------------------------------------------------------------------------- |
c45f7f84 | 105 | |
ed04d94f C |
106 | // Wrapper to video add that retry the function if there is a database error |
107 | // We need this because we run the transaction in SERIALIZABLE isolation that can fail | |
108 | function addVideoRetryWrapper (req, res, next) { | |
109 | utils.transactionRetryer( | |
110 | function (callback) { | |
111 | return addVideo(req, res, req.files.videofile[0], callback) | |
112 | }, | |
113 | function (err) { | |
114 | if (err) { | |
115 | logger.error('Cannot insert the video with many retries.', { error: err }) | |
116 | return next(err) | |
117 | } | |
118 | ||
119 | // TODO : include Location of the new video -> 201 | |
120 | return res.type('json').status(204).end() | |
121 | } | |
122 | ) | |
123 | } | |
124 | ||
125 | function addVideo (req, res, videoFile, callback) { | |
bc503c2a | 126 | const videoInfos = req.body |
9f10b292 | 127 | |
1a42c9e2 | 128 | waterfall([ |
807df9e6 | 129 | |
ed04d94f C |
130 | function startTransaction (callbackWaterfall) { |
131 | db.sequelize.transaction({ isolationLevel: 'SERIALIZABLE' }).asCallback(function (err, t) { | |
132 | return callbackWaterfall(err, t) | |
7920c273 C |
133 | }) |
134 | }, | |
135 | ||
ed04d94f | 136 | function findOrCreateAuthor (t, callbackWaterfall) { |
4712081f | 137 | const user = res.locals.oauth.token.User |
feb4bdfd | 138 | |
4ff0d862 C |
139 | const name = user.username |
140 | // null because it is OUR pod | |
141 | const podId = null | |
142 | const userId = user.id | |
4712081f | 143 | |
4ff0d862 | 144 | db.Author.findOrCreateAuthor(name, podId, userId, t, function (err, authorInstance) { |
ed04d94f | 145 | return callbackWaterfall(err, t, authorInstance) |
7920c273 C |
146 | }) |
147 | }, | |
148 | ||
ed04d94f | 149 | function findOrCreateTags (t, author, callbackWaterfall) { |
7920c273 | 150 | const tags = videoInfos.tags |
4ff0d862 C |
151 | |
152 | db.Tag.findOrCreateTags(tags, t, function (err, tagInstances) { | |
ed04d94f | 153 | return callbackWaterfall(err, t, author, tagInstances) |
feb4bdfd C |
154 | }) |
155 | }, | |
156 | ||
ed04d94f | 157 | function createVideoObject (t, author, tagInstances, callbackWaterfall) { |
807df9e6 C |
158 | const videoData = { |
159 | name: videoInfos.name, | |
558d7c23 C |
160 | remoteId: null, |
161 | extname: path.extname(videoFile.filename), | |
807df9e6 | 162 | description: videoInfos.description, |
67100f1f | 163 | duration: videoFile.duration, |
feb4bdfd | 164 | authorId: author.id |
807df9e6 C |
165 | } |
166 | ||
feb4bdfd | 167 | const video = db.Video.build(videoData) |
558d7c23 | 168 | |
ed04d94f | 169 | return callbackWaterfall(null, t, author, tagInstances, video) |
558d7c23 C |
170 | }, |
171 | ||
feb4bdfd | 172 | // Set the videoname the same as the id |
ed04d94f | 173 | function renameVideoFile (t, author, tagInstances, video, callbackWaterfall) { |
558d7c23 C |
174 | const videoDir = constants.CONFIG.STORAGE.VIDEOS_DIR |
175 | const source = path.join(videoDir, videoFile.filename) | |
f285faa0 | 176 | const destination = path.join(videoDir, video.getVideoFilename()) |
558d7c23 C |
177 | |
178 | fs.rename(source, destination, function (err) { | |
ed04d94f C |
179 | if (err) return callbackWaterfall(err) |
180 | ||
181 | // This is important in case if there is another attempt | |
182 | videoFile.filename = video.getVideoFilename() | |
183 | return callbackWaterfall(null, t, author, tagInstances, video) | |
558d7c23 C |
184 | }) |
185 | }, | |
186 | ||
ed04d94f | 187 | function insertVideoIntoDB (t, author, tagInstances, video, callbackWaterfall) { |
7920c273 C |
188 | const options = { transaction: t } |
189 | ||
190 | // Add tags association | |
191 | video.save(options).asCallback(function (err, videoCreated) { | |
ed04d94f | 192 | if (err) return callbackWaterfall(err) |
7920c273 | 193 | |
feb4bdfd C |
194 | // Do not forget to add Author informations to the created video |
195 | videoCreated.Author = author | |
196 | ||
ed04d94f | 197 | return callbackWaterfall(err, t, tagInstances, videoCreated) |
3a8a8b51 | 198 | }) |
807df9e6 C |
199 | }, |
200 | ||
ed04d94f | 201 | function associateTagsToVideo (t, tagInstances, video, callbackWaterfall) { |
7920c273 C |
202 | const options = { transaction: t } |
203 | ||
204 | video.setTags(tagInstances, options).asCallback(function (err) { | |
205 | video.Tags = tagInstances | |
206 | ||
ed04d94f | 207 | return callbackWaterfall(err, t, video) |
7920c273 C |
208 | }) |
209 | }, | |
210 | ||
ed04d94f | 211 | function sendToFriends (t, video, callbackWaterfall) { |
7b1f49de | 212 | video.toAddRemoteJSON(function (err, remoteVideo) { |
ed04d94f | 213 | if (err) return callbackWaterfall(err) |
807df9e6 | 214 | |
528a9efa | 215 | // Now we'll add the video's meta data to our friends |
ed04d94f C |
216 | friends.addVideoToFriends(remoteVideo, t, function (err) { |
217 | return callbackWaterfall(err, t) | |
218 | }) | |
528a9efa | 219 | }) |
807df9e6 C |
220 | } |
221 | ||
7b1f49de C |
222 | ], function andFinally (err, t) { |
223 | if (err) { | |
ed04d94f C |
224 | // This is just a debug because we will retry the insert |
225 | logger.debug('Cannot insert the video.', { error: err }) | |
7b1f49de C |
226 | |
227 | // Abort transaction? | |
228 | if (t) t.rollback() | |
229 | ||
ed04d94f | 230 | return callback(err) |
7b1f49de C |
231 | } |
232 | ||
233 | // Commit transaction | |
dea32aac C |
234 | t.commit().asCallback(function (err) { |
235 | if (err) return callback(err) | |
7b1f49de | 236 | |
dea32aac C |
237 | logger.info('Video with name %s created.', videoInfos.name) |
238 | return callback(null) | |
239 | }) | |
7b1f49de C |
240 | }) |
241 | } | |
242 | ||
ed04d94f C |
243 | function updateVideoRetryWrapper (req, res, next) { |
244 | utils.transactionRetryer( | |
245 | function (callback) { | |
246 | return updateVideo(req, res, callback) | |
247 | }, | |
248 | function (err) { | |
249 | if (err) { | |
250 | logger.error('Cannot update the video with many retries.', { error: err }) | |
251 | return next(err) | |
252 | } | |
253 | ||
254 | // TODO : include Location of the new video -> 201 | |
255 | return res.type('json').status(204).end() | |
256 | } | |
257 | ) | |
258 | } | |
259 | ||
260 | function updateVideo (req, res, finalCallback) { | |
818f7987 | 261 | const videoInstance = res.locals.video |
7b1f49de C |
262 | const videoInfosToUpdate = req.body |
263 | ||
264 | waterfall([ | |
265 | ||
266 | function startTransaction (callback) { | |
267 | db.sequelize.transaction().asCallback(function (err, t) { | |
268 | return callback(err, t) | |
269 | }) | |
270 | }, | |
271 | ||
272 | function findOrCreateTags (t, callback) { | |
273 | if (videoInfosToUpdate.tags) { | |
274 | db.Tag.findOrCreateTags(videoInfosToUpdate.tags, t, function (err, tagInstances) { | |
275 | return callback(err, t, tagInstances) | |
276 | }) | |
277 | } else { | |
278 | return callback(null, t, null) | |
279 | } | |
280 | }, | |
281 | ||
282 | function updateVideoIntoDB (t, tagInstances, callback) { | |
283 | const options = { transaction: t } | |
284 | ||
285 | if (videoInfosToUpdate.name) videoInstance.set('name', videoInfosToUpdate.name) | |
286 | if (videoInfosToUpdate.description) videoInstance.set('description', videoInfosToUpdate.description) | |
287 | ||
288 | // Add tags association | |
289 | videoInstance.save(options).asCallback(function (err) { | |
7b1f49de C |
290 | return callback(err, t, tagInstances) |
291 | }) | |
292 | }, | |
293 | ||
294 | function associateTagsToVideo (t, tagInstances, callback) { | |
295 | if (tagInstances) { | |
296 | const options = { transaction: t } | |
297 | ||
298 | videoInstance.setTags(tagInstances, options).asCallback(function (err) { | |
299 | videoInstance.Tags = tagInstances | |
300 | ||
301 | return callback(err, t) | |
302 | }) | |
303 | } else { | |
304 | return callback(null, t) | |
305 | } | |
306 | }, | |
307 | ||
308 | function sendToFriends (t, callback) { | |
309 | const json = videoInstance.toUpdateRemoteJSON() | |
310 | ||
311 | // Now we'll update the video's meta data to our friends | |
ed04d94f C |
312 | friends.updateVideoToFriends(json, t, function (err) { |
313 | return callback(err, t) | |
314 | }) | |
7b1f49de C |
315 | } |
316 | ||
7920c273 | 317 | ], function andFinally (err, t) { |
807df9e6 | 318 | if (err) { |
ed04d94f | 319 | logger.debug('Cannot update the video.', { error: err }) |
7920c273 C |
320 | |
321 | // Abort transaction? | |
322 | if (t) t.rollback() | |
323 | ||
ed04d94f | 324 | return finalCallback(err) |
807df9e6 C |
325 | } |
326 | ||
7920c273 | 327 | // Commit transaction |
dea32aac C |
328 | t.commit().asCallback(function (err) { |
329 | if (err) return finalCallback(err) | |
7920c273 | 330 | |
dea32aac C |
331 | logger.info('Video with name %s updated.', videoInfosToUpdate.name) |
332 | return finalCallback(null) | |
333 | }) | |
9f10b292 C |
334 | }) |
335 | } | |
8c308c2b | 336 | |
68ce3ae0 | 337 | function getVideo (req, res, next) { |
818f7987 C |
338 | const videoInstance = res.locals.video |
339 | res.json(videoInstance.toFormatedJSON()) | |
9f10b292 | 340 | } |
8c308c2b | 341 | |
9f10b292 | 342 | function listVideos (req, res, next) { |
feb4bdfd | 343 | db.Video.listForApi(req.query.start, req.query.count, req.query.sort, function (err, videosList, videosTotal) { |
9f10b292 | 344 | if (err) return next(err) |
c45f7f84 | 345 | |
55fa55a9 | 346 | res.json(utils.getFormatedObjects(videosList, videosTotal)) |
9f10b292 C |
347 | }) |
348 | } | |
c45f7f84 | 349 | |
9f10b292 | 350 | function removeVideo (req, res, next) { |
818f7987 | 351 | const videoInstance = res.locals.video |
8c308c2b | 352 | |
818f7987 | 353 | videoInstance.destroy().asCallback(function (err) { |
807df9e6 C |
354 | if (err) { |
355 | logger.error('Errors when removed the video.', { error: err }) | |
356 | return next(err) | |
357 | } | |
358 | ||
359 | return res.type('json').status(204).end() | |
9f10b292 C |
360 | }) |
361 | } | |
8c308c2b | 362 | |
9f10b292 | 363 | function searchVideos (req, res, next) { |
7920c273 C |
364 | db.Video.searchAndPopulateAuthorAndPodAndTags( |
365 | req.params.value, req.query.field, req.query.start, req.query.count, req.query.sort, | |
366 | function (err, videosList, videosTotal) { | |
367 | if (err) return next(err) | |
8c308c2b | 368 | |
55fa55a9 | 369 | res.json(utils.getFormatedObjects(videosList, videosTotal)) |
7920c273 C |
370 | } |
371 | ) | |
9f10b292 | 372 | } |
c173e565 | 373 | |
55fa55a9 C |
374 | function listVideoAbuses (req, res, next) { |
375 | db.VideoAbuse.listForApi(req.query.start, req.query.count, req.query.sort, function (err, abusesList, abusesTotal) { | |
376 | if (err) return next(err) | |
2df82d42 | 377 | |
55fa55a9 | 378 | res.json(utils.getFormatedObjects(abusesList, abusesTotal)) |
2df82d42 | 379 | }) |
55fa55a9 | 380 | } |
2df82d42 | 381 | |
bf4ff8fe C |
382 | function reportVideoAbuseRetryWrapper (req, res, next) { |
383 | utils.transactionRetryer( | |
384 | function (callback) { | |
385 | return reportVideoAbuse(req, res, callback) | |
386 | }, | |
387 | function (err) { | |
388 | if (err) { | |
389 | logger.error('Cannot report abuse to the video with many retries.', { error: err }) | |
390 | return next(err) | |
391 | } | |
392 | ||
393 | return res.type('json').status(204).end() | |
394 | } | |
395 | ) | |
396 | } | |
397 | ||
398 | function reportVideoAbuse (req, res, finalCallback) { | |
55fa55a9 C |
399 | const videoInstance = res.locals.video |
400 | const reporterUsername = res.locals.oauth.token.User.username | |
401 | ||
402 | const abuse = { | |
403 | reporterUsername, | |
404 | reason: req.body.reason, | |
405 | videoId: videoInstance.id, | |
406 | reporterPodId: null // This is our pod that reported this abuse | |
68ce3ae0 | 407 | } |
55fa55a9 | 408 | |
bf4ff8fe C |
409 | waterfall([ |
410 | ||
411 | function startTransaction (callback) { | |
412 | db.sequelize.transaction().asCallback(function (err, t) { | |
413 | return callback(err, t) | |
414 | }) | |
415 | }, | |
416 | ||
417 | function createAbuse (t, callback) { | |
418 | db.VideoAbuse.create(abuse).asCallback(function (err, abuse) { | |
419 | return callback(err, t, abuse) | |
420 | }) | |
421 | }, | |
422 | ||
423 | function sendToFriendsIfNeeded (t, abuse, callback) { | |
424 | // We send the information to the destination pod | |
425 | if (videoInstance.isOwned() === false) { | |
426 | const reportData = { | |
427 | reporterUsername, | |
428 | reportReason: abuse.reason, | |
429 | videoRemoteId: videoInstance.remoteId | |
430 | } | |
55fa55a9 | 431 | |
bf4ff8fe | 432 | friends.reportAbuseVideoToFriend(reportData, videoInstance) |
55fa55a9 C |
433 | } |
434 | ||
bf4ff8fe | 435 | return callback(null, t) |
55fa55a9 C |
436 | } |
437 | ||
bf4ff8fe C |
438 | ], function andFinally (err, t) { |
439 | if (err) { | |
440 | logger.debug('Cannot update the video.', { error: err }) | |
441 | ||
442 | // Abort transaction? | |
443 | if (t) t.rollback() | |
444 | ||
445 | return finalCallback(err) | |
446 | } | |
447 | ||
448 | // Commit transaction | |
dea32aac C |
449 | t.commit().asCallback(function (err) { |
450 | if (err) return finalCallback(err) | |
bf4ff8fe | 451 | |
dea32aac C |
452 | logger.info('Abuse report for video %s created.', videoInstance.name) |
453 | return finalCallback(null) | |
454 | }) | |
55fa55a9 | 455 | }) |
2df82d42 | 456 | } |
55fa55a9 | 457 |