PeerTube/server/controllers/api/v1/videos.js

291 lines
8.0 KiB
JavaScript
Raw Normal View History

'use strict'
const async = require('async')
2016-03-16 16:29:27 -05:00
const config = require('config')
const express = require('express')
const fs = require('fs')
const path = require('path')
2016-03-16 16:29:27 -05:00
const multer = require('multer')
const constants = require('../../../initializers/constants')
2016-03-16 16:29:27 -05:00
const logger = require('../../../helpers/logger')
const friends = require('../../../lib/friends')
const middlewares = require('../../../middlewares')
const oAuth2 = middlewares.oauth2
const pagination = middlewares.pagination
const reqValidator = middlewares.reqValidators
const reqValidatorPagination = reqValidator.pagination
2016-05-17 14:03:00 -05:00
const reqValidatorSort = reqValidator.sort
const reqValidatorVideos = reqValidator.videos
const search = middlewares.search
2016-05-17 14:03:00 -05:00
const sort = middlewares.sort
const utils = require('../../../helpers/utils')
2016-03-16 16:29:27 -05:00
const Videos = require('../../../models/videos') // model
const videos = require('../../../lib/videos')
const webtorrent = require('../../../lib/webtorrent')
const router = express.Router()
const uploads = config.get('storage.uploads')
// multer configuration
2016-03-16 16:29:27 -05:00
const storage = multer.diskStorage({
destination: function (req, file, cb) {
cb(null, uploads)
},
filename: function (req, file, cb) {
2016-03-16 16:29:27 -05:00
let extension = ''
if (file.mimetype === 'video/webm') extension = 'webm'
else if (file.mimetype === 'video/mp4') extension = 'mp4'
else if (file.mimetype === 'video/ogg') extension = 'ogv'
utils.generateRandomString(16, function (err, randomString) {
const fieldname = err ? undefined : randomString
cb(null, fieldname + '.' + extension)
})
}
})
const reqFiles = multer({ storage: storage }).fields([{ name: 'videofile', maxCount: 1 }])
const thumbnailsDir = path.join(__dirname, '..', '..', '..', '..', config.get('storage.thumbnails'))
2015-06-09 10:41:40 -05:00
router.get('/',
reqValidatorPagination.pagination,
2016-05-17 14:03:00 -05:00
reqValidatorSort.videosSort,
sort.setVideosSort,
pagination.setPagination,
listVideos
)
router.post('/',
oAuth2.authenticate,
reqFiles,
reqValidatorVideos.videosAdd,
addVideo
)
router.get('/:id',
reqValidatorVideos.videosGet,
getVideo
)
router.delete('/:id',
oAuth2.authenticate,
reqValidatorVideos.videosRemove,
removeVideo
)
router.get('/search/:value',
reqValidatorVideos.videosSearch,
reqValidatorPagination.pagination,
2016-05-17 14:03:00 -05:00
reqValidatorSort.videosSort,
sort.setVideosSort,
pagination.setPagination,
search.setVideosSearch,
searchVideos
)
2015-06-09 10:41:40 -05:00
// ---------------------------------------------------------------------------
2016-01-31 04:23:52 -06:00
module.exports = router
2016-01-31 04:23:52 -06:00
// ---------------------------------------------------------------------------
2016-01-31 04:23:52 -06:00
function addVideo (req, res, next) {
const videoFile = req.files.videofile[0]
const videoInfos = req.body
async.waterfall([
2016-05-16 12:49:10 -05:00
function seedTheVideo (callback) {
videos.seed(videoFile.path, callback)
},
2015-06-09 10:41:40 -05:00
2016-05-16 12:49:10 -05:00
function createThumbnail (torrent, callback) {
videos.createVideoThumbnail(videoFile.path, function (err, thumbnailName) {
2016-05-03 15:41:46 -05:00
if (err) {
// TODO: unseed the video
logger.error('Cannot make a thumbnail of the video file.')
return callback(err)
2016-05-03 15:41:46 -05:00
}
2016-05-16 12:49:10 -05:00
callback(null, torrent, thumbnailName)
})
},
2016-05-16 12:49:10 -05:00
function insertIntoDB (torrent, thumbnailName, callback) {
const videoData = {
name: videoInfos.name,
namePath: videoFile.filename,
description: videoInfos.description,
magnetUri: torrent.magnetURI,
author: res.locals.oauth.token.user.username,
2016-05-16 12:49:10 -05:00
duration: videoFile.duration,
thumbnail: thumbnailName
}
Videos.add(videoData, function (err, insertedVideo) {
if (err) {
// TODO unseed the video
// TODO remove thumbnail
logger.error('Cannot insert this video in the database.')
return callback(err)
}
2016-05-03 15:41:46 -05:00
2016-05-16 12:49:10 -05:00
return callback(null, torrent, thumbnailName, videoData, insertedVideo)
2016-05-03 15:41:46 -05:00
})
},
2016-05-16 12:49:10 -05:00
function getThumbnailBase64 (torrent, thumbnailName, videoData, insertedVideo, callback) {
videoData.createdDate = insertedVideo.createdDate
fs.readFile(thumbnailsDir + thumbnailName, function (err, thumbnailData) {
if (err) {
// TODO unseed the video
// TODO remove thumbnail
// TODO: remove video
logger.error('Cannot read the thumbnail of the video')
return callback(err)
}
return callback(null, videoData, thumbnailData)
})
},
function sendToFriends (videoData, thumbnailData, callback) {
// Set the image in base64
videoData.thumbnailBase64 = new Buffer(thumbnailData).toString('base64')
// Now we'll add the video's meta data to our friends
friends.addVideoToFriends(videoData)
return callback(null)
}
], function (err) {
if (err) {
logger.error('Cannot insert the video.')
return next(err)
}
// TODO : include Location of the new video -> 201
return res.type('json').status(204).end()
})
}
2015-06-09 10:41:40 -05:00
function getVideo (req, res, next) {
Videos.get(req.params.id, function (err, videoObj) {
if (err) return next(err)
2015-06-09 10:41:40 -05:00
const state = videos.getVideoState(videoObj)
2016-03-18 10:28:09 -05:00
if (state.exist === false) {
return res.type('json').status(204).end()
}
2015-06-09 10:41:40 -05:00
res.json(getFormatedVideo(videoObj))
})
}
2015-06-09 10:41:40 -05:00
function listVideos (req, res, next) {
Videos.list(req.query.start, req.query.count, req.query.sort, function (err, videosList, totalVideos) {
if (err) return next(err)
2016-01-31 04:23:52 -06:00
res.json(getFormatedVideos(videosList, totalVideos))
})
}
2016-01-31 04:23:52 -06:00
function removeVideo (req, res, next) {
const videoId = req.params.id
2015-06-09 10:41:40 -05:00
async.waterfall([
function getVideo (callback) {
Videos.get(videoId, callback)
},
function removeVideoTorrent (video, callback) {
removeTorrent(video.magnetUri, function () {
return callback(null, video)
})
},
function removeFromDB (video, callback) {
Videos.removeOwned(req.params.id, function (err) {
if (err) return callback(err)
2016-02-04 14:10:33 -06:00
return callback(null, video)
})
},
function removeVideoData (video, callback) {
videos.removeVideosDataFromDisk([ video ], function (err) {
if (err) logger.error('Cannot remove video data from disk.', { video: video })
2016-02-04 14:10:33 -06:00
return callback(null, video)
2016-02-04 14:10:33 -06:00
})
},
function sendInformationToFriends (video, callback) {
const params = {
name: video.name,
magnetUri: video.magnetUri
}
friends.removeVideoToFriends(params)
return callback(null)
}
], function (err) {
if (err) {
logger.error('Errors when removed the video.', { error: err })
return next(err)
}
return res.type('json').status(204).end()
})
}
2015-06-09 10:41:40 -05:00
function searchVideos (req, res, next) {
Videos.search(req.params.value, req.query.field, req.query.start, req.query.count, req.query.sort,
function (err, videosList, totalVideos) {
if (err) return next(err)
2015-06-09 10:41:40 -05:00
res.json(getFormatedVideos(videosList, totalVideos))
})
}
2016-02-04 14:10:33 -06:00
// ---------------------------------------------------------------------------
2016-02-04 14:10:33 -06:00
function getFormatedVideo (videoObj) {
const formatedVideo = {
id: videoObj._id,
name: videoObj.name,
description: videoObj.description,
podUrl: videoObj.podUrl.replace(/^https?:\/\//, ''),
isLocal: videos.getVideoState(videoObj).owned,
magnetUri: videoObj.magnetUri,
author: videoObj.author,
duration: videoObj.duration,
2016-05-13 13:42:11 -05:00
thumbnailPath: constants.THUMBNAILS_STATIC_PATH + '/' + videoObj.thumbnail,
createdDate: videoObj.createdDate
2016-03-18 10:28:09 -05:00
}
return formatedVideo
2016-03-18 10:28:09 -05:00
}
function getFormatedVideos (videosObj, totalVideos) {
const formatedVideos = []
2016-03-18 10:28:09 -05:00
videosObj.forEach(function (videoObj) {
formatedVideos.push(getFormatedVideo(videoObj))
2016-03-18 10:28:09 -05:00
})
return {
total: totalVideos,
data: formatedVideos
}
2016-03-18 10:28:09 -05:00
}
// Maybe the torrent is not seeded, but we catch the error to don't stop the removing process
function removeTorrent (magnetUri, callback) {
try {
webtorrent.remove(magnetUri, callback)
} catch (err) {
logger.warn('Cannot remove the torrent from WebTorrent', { err: err })
return callback(null)
2016-02-04 14:10:33 -06:00
}
}