Improve torrent/video download
This commit is contained in:
parent
a20776fcbb
commit
02756fbd11
5 changed files with 61 additions and 6 deletions
|
@ -41,7 +41,7 @@ export class VideoDownloadComponent implements OnInit {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
const link = this.downloadType === 'direct' ? file.fileUrl : file.torrentUrl
|
const link = this.downloadType === 'direct' ? file.fileDownloadUrl : file.torrentDownloadUrl
|
||||||
window.open(link)
|
window.location.assign(link)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,8 +1,9 @@
|
||||||
import * as cors from 'cors'
|
import * as cors from 'cors'
|
||||||
import * as express from 'express'
|
import * as express from 'express'
|
||||||
import { CONFIG, STATIC_MAX_AGE, STATIC_PATHS } from '../initializers'
|
import { CONFIG, STATIC_DOWNLOAD_PATHS, STATIC_MAX_AGE, STATIC_PATHS } from '../initializers'
|
||||||
import { VideosPreviewCache } from '../lib/cache'
|
import { VideosPreviewCache } from '../lib/cache'
|
||||||
import { asyncMiddleware } from '../middlewares'
|
import { asyncMiddleware, videosGetValidator } from '../middlewares'
|
||||||
|
import { VideoModel } from '../models/video/video'
|
||||||
|
|
||||||
const staticRouter = express.Router()
|
const staticRouter = express.Router()
|
||||||
|
|
||||||
|
@ -16,6 +17,11 @@ staticRouter.use(
|
||||||
cors(),
|
cors(),
|
||||||
express.static(torrentsPhysicalPath, { maxAge: 0 }) // Don't cache because we could regenerate the torrent file
|
express.static(torrentsPhysicalPath, { maxAge: 0 }) // Don't cache because we could regenerate the torrent file
|
||||||
)
|
)
|
||||||
|
staticRouter.use(
|
||||||
|
STATIC_DOWNLOAD_PATHS.TORRENTS + ':id-:resolution([0-9]+).torrent',
|
||||||
|
asyncMiddleware(videosGetValidator),
|
||||||
|
asyncMiddleware(downloadTorrent)
|
||||||
|
)
|
||||||
|
|
||||||
// Videos path for webseeding
|
// Videos path for webseeding
|
||||||
const videosPhysicalPath = CONFIG.STORAGE.VIDEOS_DIR
|
const videosPhysicalPath = CONFIG.STORAGE.VIDEOS_DIR
|
||||||
|
@ -24,6 +30,11 @@ staticRouter.use(
|
||||||
cors(),
|
cors(),
|
||||||
express.static(videosPhysicalPath, { maxAge: STATIC_MAX_AGE })
|
express.static(videosPhysicalPath, { maxAge: STATIC_MAX_AGE })
|
||||||
)
|
)
|
||||||
|
staticRouter.use(
|
||||||
|
STATIC_DOWNLOAD_PATHS.VIDEOS + ':id-:resolution([0-9]+).:extension',
|
||||||
|
asyncMiddleware(videosGetValidator),
|
||||||
|
asyncMiddleware(downloadVideoFile)
|
||||||
|
)
|
||||||
|
|
||||||
// Thumbnails path for express
|
// Thumbnails path for express
|
||||||
const thumbnailsPhysicalPath = CONFIG.STORAGE.THUMBNAILS_DIR
|
const thumbnailsPhysicalPath = CONFIG.STORAGE.THUMBNAILS_DIR
|
||||||
|
@ -64,3 +75,26 @@ async function getPreview (req: express.Request, res: express.Response, next: ex
|
||||||
|
|
||||||
return res.sendFile(path, { maxAge: STATIC_MAX_AGE })
|
return res.sendFile(path, { maxAge: STATIC_MAX_AGE })
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async function downloadTorrent (req: express.Request, res: express.Response, next: express.NextFunction) {
|
||||||
|
const { video, videoFile } = getVideoAndFileOr404(req, res)
|
||||||
|
if (!videoFile) return res.status(404).end()
|
||||||
|
|
||||||
|
return res.download(video.getTorrentFilePath(videoFile), `${video.name}-${videoFile.resolution}p.torrent`)
|
||||||
|
}
|
||||||
|
|
||||||
|
async function downloadVideoFile (req: express.Request, res: express.Response, next: express.NextFunction) {
|
||||||
|
const { video, videoFile } = getVideoAndFileOr404(req, res)
|
||||||
|
if (!videoFile) return res.status(404).end()
|
||||||
|
|
||||||
|
return res.download(video.getVideoFilePath(videoFile), `${video.name}-${videoFile.resolution}p${videoFile.extname}`)
|
||||||
|
}
|
||||||
|
|
||||||
|
function getVideoAndFileOr404 (req: express.Request, res: express.Response) {
|
||||||
|
const resolution = parseInt(req.params.resolution, 10)
|
||||||
|
const video: VideoModel = res.locals.video
|
||||||
|
|
||||||
|
const videoFile = video.VideoFiles.find(f => f.resolution === resolution)
|
||||||
|
|
||||||
|
return { video, videoFile }
|
||||||
|
}
|
||||||
|
|
|
@ -389,6 +389,10 @@ const STATIC_PATHS = {
|
||||||
WEBSEED: '/static/webseed/',
|
WEBSEED: '/static/webseed/',
|
||||||
AVATARS: '/static/avatars/'
|
AVATARS: '/static/avatars/'
|
||||||
}
|
}
|
||||||
|
const STATIC_DOWNLOAD_PATHS = {
|
||||||
|
TORRENTS: '/download/torrents/',
|
||||||
|
VIDEOS: '/download/videos/'
|
||||||
|
}
|
||||||
|
|
||||||
// Cache control
|
// Cache control
|
||||||
let STATIC_MAX_AGE = '30d'
|
let STATIC_MAX_AGE = '30d'
|
||||||
|
@ -493,6 +497,7 @@ export {
|
||||||
USER_PASSWORD_RESET_LIFETIME,
|
USER_PASSWORD_RESET_LIFETIME,
|
||||||
IMAGE_MIMETYPE_EXT,
|
IMAGE_MIMETYPE_EXT,
|
||||||
SCHEDULER_INTERVAL,
|
SCHEDULER_INTERVAL,
|
||||||
|
STATIC_DOWNLOAD_PATHS,
|
||||||
RATES_LIMIT,
|
RATES_LIMIT,
|
||||||
JOB_COMPLETED_LIFETIME,
|
JOB_COMPLETED_LIFETIME,
|
||||||
VIDEO_VIEW_LIFETIME
|
VIDEO_VIEW_LIFETIME
|
||||||
|
|
|
@ -29,7 +29,6 @@ import { VideoPrivacy, VideoResolution } from '../../../shared'
|
||||||
import { VideoTorrentObject } from '../../../shared/models/activitypub/objects'
|
import { VideoTorrentObject } from '../../../shared/models/activitypub/objects'
|
||||||
import { Video, VideoDetails, VideoFile } from '../../../shared/models/videos'
|
import { Video, VideoDetails, VideoFile } from '../../../shared/models/videos'
|
||||||
import { VideoFilter } from '../../../shared/models/videos/video-query.type'
|
import { VideoFilter } from '../../../shared/models/videos/video-query.type'
|
||||||
import { activityPubCollectionPagination } from '../../helpers/activitypub'
|
|
||||||
import {
|
import {
|
||||||
createTorrentPromise,
|
createTorrentPromise,
|
||||||
peertubeTruncate,
|
peertubeTruncate,
|
||||||
|
@ -59,6 +58,7 @@ import {
|
||||||
CONSTRAINTS_FIELDS,
|
CONSTRAINTS_FIELDS,
|
||||||
PREVIEWS_SIZE,
|
PREVIEWS_SIZE,
|
||||||
REMOTE_SCHEME,
|
REMOTE_SCHEME,
|
||||||
|
STATIC_DOWNLOAD_PATHS,
|
||||||
STATIC_PATHS,
|
STATIC_PATHS,
|
||||||
THUMBNAILS_SIZE,
|
THUMBNAILS_SIZE,
|
||||||
VIDEO_CATEGORIES,
|
VIDEO_CATEGORIES,
|
||||||
|
@ -979,6 +979,10 @@ export class VideoModel extends Model<VideoModel> {
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
getTorrentFilePath (videoFile: VideoFileModel) {
|
||||||
|
return join(CONFIG.STORAGE.TORRENTS_DIR, this.getTorrentFileName(videoFile))
|
||||||
|
}
|
||||||
|
|
||||||
getVideoFilePath (videoFile: VideoFileModel) {
|
getVideoFilePath (videoFile: VideoFileModel) {
|
||||||
return join(CONFIG.STORAGE.VIDEOS_DIR, this.getVideoFilename(videoFile))
|
return join(CONFIG.STORAGE.VIDEOS_DIR, this.getVideoFilename(videoFile))
|
||||||
}
|
}
|
||||||
|
@ -1112,7 +1116,9 @@ export class VideoModel extends Model<VideoModel> {
|
||||||
magnetUri: this.generateMagnetUri(videoFile, baseUrlHttp, baseUrlWs),
|
magnetUri: this.generateMagnetUri(videoFile, baseUrlHttp, baseUrlWs),
|
||||||
size: videoFile.size,
|
size: videoFile.size,
|
||||||
torrentUrl: this.getTorrentUrl(videoFile, baseUrlHttp),
|
torrentUrl: this.getTorrentUrl(videoFile, baseUrlHttp),
|
||||||
fileUrl: this.getVideoFileUrl(videoFile, baseUrlHttp)
|
torrentDownloadUrl: this.getTorrentDownloadUrl(videoFile, baseUrlHttp),
|
||||||
|
fileUrl: this.getVideoFileUrl(videoFile, baseUrlHttp),
|
||||||
|
fileDownloadUrl: this.getVideoFileDownloadUrl(videoFile, baseUrlHttp)
|
||||||
} as VideoFile
|
} as VideoFile
|
||||||
})
|
})
|
||||||
.sort((a, b) => {
|
.sort((a, b) => {
|
||||||
|
@ -1367,10 +1373,18 @@ export class VideoModel extends Model<VideoModel> {
|
||||||
return baseUrlHttp + STATIC_PATHS.TORRENTS + this.getTorrentFileName(videoFile)
|
return baseUrlHttp + STATIC_PATHS.TORRENTS + this.getTorrentFileName(videoFile)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private getTorrentDownloadUrl (videoFile: VideoFileModel, baseUrlHttp: string) {
|
||||||
|
return baseUrlHttp + STATIC_DOWNLOAD_PATHS.TORRENTS + this.getTorrentFileName(videoFile)
|
||||||
|
}
|
||||||
|
|
||||||
private getVideoFileUrl (videoFile: VideoFileModel, baseUrlHttp: string) {
|
private getVideoFileUrl (videoFile: VideoFileModel, baseUrlHttp: string) {
|
||||||
return baseUrlHttp + STATIC_PATHS.WEBSEED + this.getVideoFilename(videoFile)
|
return baseUrlHttp + STATIC_PATHS.WEBSEED + this.getVideoFilename(videoFile)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private getVideoFileDownloadUrl (videoFile: VideoFileModel, baseUrlHttp: string) {
|
||||||
|
return baseUrlHttp + STATIC_DOWNLOAD_PATHS.VIDEOS + this.getVideoFilename(videoFile)
|
||||||
|
}
|
||||||
|
|
||||||
private generateMagnetUri (videoFile: VideoFileModel, baseUrlHttp: string, baseUrlWs: string) {
|
private generateMagnetUri (videoFile: VideoFileModel, baseUrlHttp: string, baseUrlWs: string) {
|
||||||
const xs = this.getTorrentUrl(videoFile, baseUrlHttp)
|
const xs = this.getTorrentUrl(videoFile, baseUrlHttp)
|
||||||
const announce = [ baseUrlWs + '/tracker/socket', baseUrlHttp + '/tracker/announce' ]
|
const announce = [ baseUrlWs + '/tracker/socket', baseUrlHttp + '/tracker/announce' ]
|
||||||
|
|
|
@ -14,7 +14,9 @@ export interface VideoFile {
|
||||||
resolution: VideoConstant<VideoResolution>
|
resolution: VideoConstant<VideoResolution>
|
||||||
size: number // Bytes
|
size: number // Bytes
|
||||||
torrentUrl: string
|
torrentUrl: string
|
||||||
|
torrentDownloadUrl: string
|
||||||
fileUrl: string
|
fileUrl: string
|
||||||
|
fileDownloadUrl: string
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface Video {
|
export interface Video {
|
||||||
|
|
Loading…
Add table
Reference in a new issue