Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add support for saving video files to object storage #4290

Merged
merged 24 commits into from
Aug 17, 2021
Merged
Show file tree
Hide file tree
Changes from 19 commits
Commits
Show all changes
24 commits
Select commit Hold shift + click to select a range
4145ba0
Add support for saving video files to object storage
pingiun Jul 30, 2021
4f5d19b
Add support for custom url generation on s3 stored files
pingiun Aug 1, 2021
67f7339
Upload files to s3 concurrently and delete originals afterwards
pingiun Aug 1, 2021
340dc0c
Only publish after move to object storage is complete
pingiun Aug 1, 2021
9432222
Use base url instead of url template
pingiun Aug 2, 2021
70e1ef7
Fix mistyped config field
pingiun Aug 3, 2021
121f8b5
Add rudenmentary way to download before transcode
pingiun Aug 3, 2021
2374244
Implement Chocobozzz suggestions
pingiun Aug 3, 2021
7d788e7
Import correct function
pingiun Aug 3, 2021
14fb34b
Support multipart upload
pingiun Aug 3, 2021
9422fd9
Remove import of node 15.0 module stream/promises
pingiun Aug 6, 2021
09cfb18
Extend maximum upload job length
pingiun Aug 6, 2021
25d6608
Use dynamic part size for really large uploads
pingiun Aug 6, 2021
6b5c9ca
Fix decreasePendingMove query
pingiun Aug 6, 2021
2a26bb7
Resolve various PR comments
pingiun Aug 6, 2021
24c45a5
Move to object storage after optimize
pingiun Aug 6, 2021
50d6556
Make upload size configurable and increase default
pingiun Aug 7, 2021
a54e28a
Prune webtorrent files that are stored in object storage
pingiun Aug 9, 2021
72eb147
Move files after transcoding jobs
Chocobozzz Aug 12, 2021
b884938
Merge branch 'develop' into add-object-storage-support
Chocobozzz Aug 13, 2021
831f973
Fix federation
Chocobozzz Aug 13, 2021
2b71af9
Add video path manager
Chocobozzz Aug 16, 2021
ffa0c8c
Support move to external storage job in client
Chocobozzz Aug 16, 2021
6b7c3bc
Fix live object storage tests
Chocobozzz Aug 16, 2021
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
5 changes: 5 additions & 0 deletions .github/workflows/test.yml
Original file line number Diff line number Diff line change
Expand Up @@ -31,6 +31,11 @@ jobs:
ports:
- 10389:10389

s3ninja:
image: scireum/s3-ninja
ports:
- 9444:9000

strategy:
fail-fast: false
matrix:
Expand Down
31 changes: 31 additions & 0 deletions config/default.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -95,6 +95,37 @@ storage:
# If not, peertube will fallback to the default fil
client_overrides: 'storage/client-overrides/'

object_storage:
enabled: false

# Without protocol, will default to HTTPS
endpoint: 's3.amazonaws.com'

region: 'us-east-1'

credentials:
access_key_id: 'access-key'
Chocobozzz marked this conversation as resolved.
Show resolved Hide resolved
secret_access_key: 'secret-access-key'

# Maximum amount to upload in one request to object storage
max_upload_part: 2MB
Chocobozzz marked this conversation as resolved.
Show resolved Hide resolved

streaming_playlists:
bucket_name: 'streaming-playlists'

# Allows setting all buckets to the same value but with a different prefix
prefix: '' # Example: 'streaming-playlists:'

# Base url for object URL generation, scheme and host will be replaced by this URL
# Useful when you want to use a CDN/external proxy
base_url: '' # Example: 'https://mirror.example.com'

# Same settings but for webtorrent videos
videos:
bucket_name: 'videos'
prefix: ''
base_url: ''

log:
level: 'info' # 'debug' | 'info' | 'warn' | 'error'
rotation:
Expand Down
1 change: 1 addition & 0 deletions package.json
Original file line number Diff line number Diff line change
Expand Up @@ -73,6 +73,7 @@
"swagger-cli": "swagger-cli"
},
"dependencies": {
"@aws-sdk/client-s3": "^3.23.0",
"@uploadx/core": "^4.4.0",
"async": "^3.0.1",
"async-lru": "^1.1.1",
Expand Down
3 changes: 2 additions & 1 deletion scripts/ci.sh
Original file line number Diff line number Diff line change
Expand Up @@ -89,9 +89,10 @@ elif [ "$1" = "api-4" ]; then

moderationFiles=$(findTestFiles ./dist/server/tests/api/moderation)
redundancyFiles=$(findTestFiles ./dist/server/tests/api/redundancy)
objectStorageFiles=$(findTestFiles ./dist/server/tests/api/object-storage)
activitypubFiles=$(findTestFiles ./dist/server/tests/api/activitypub)

MOCHA_PARALLEL=true TS_NODE_FILES=true runTest "$1" 2 $moderationFiles $redundancyFiles $activitypubFiles
MOCHA_PARALLEL=true TS_NODE_FILES=true runTest "$1" 2 $moderationFiles $redundancyFiles $activitypubFiles $objectStorageFiles
elif [ "$1" = "external-plugins" ]; then
npm run build:server

Expand Down
3 changes: 2 additions & 1 deletion scripts/create-transcoding-job.ts
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,7 @@ import { computeResolutionsToTranscode } from '@server/helpers/ffprobe-utils'
import { VideoTranscodingPayload } from '@shared/models'
import { CONFIG } from '@server/initializers/config'
import { isUUIDValid } from '@server/helpers/custom-validators/misc'
import { addTranscodingJob } from '@server/lib/video'

program
.option('-v, --video [videoUUID]', 'Video UUID')
Expand Down Expand Up @@ -90,7 +91,7 @@ async function run () {
await JobQueue.Instance.init()

for (const d of dataInput) {
await JobQueue.Instance.createJobWithPromise({ type: 'video-transcoding', payload: d })
await addTranscodingJob(d, {})
console.log('Transcoding job for video %s created.', video.uuid)
}
}
27 changes: 17 additions & 10 deletions scripts/optimize-old-videos.ts
Original file line number Diff line number Diff line change
@@ -1,15 +1,18 @@
import { registerTSPaths } from '../server/helpers/register-ts-paths'
registerTSPaths()

import { getDurationFromVideoFile, getVideoFileBitrate, getVideoFileFPS, getVideoFileResolution } from '../server/helpers/ffprobe-utils'
import { VideoModel } from '../server/models/video/video'
import { optimizeOriginalVideofile } from '../server/lib/transcoding/video-transcoding'
import { initDatabaseModels } from '../server/initializers/database'
import { basename, dirname } from 'path'
import { copy, move, remove } from 'fs-extra'
import { basename, dirname } from 'path'
import { createTorrentAndSetInfoHash } from '@server/helpers/webtorrent'
import { getVideoFilePath } from '@server/lib/video-paths'
import { CONFIG } from '@server/initializers/config'
import { processMoveToObjectStorage } from '@server/lib/job-queue/handlers/move-to-object-storage'
import { getVideoFilePath, getVideoFilePathMakeAvailable } from '@server/lib/video-paths'
import { getMaxBitrate } from '@shared/core-utils'
import { MoveObjectStoragePayload } from '@shared/models'
import { getDurationFromVideoFile, getVideoFileBitrate, getVideoFileFPS, getVideoFileResolution } from '../server/helpers/ffprobe-utils'
import { registerTSPaths } from '../server/helpers/register-ts-paths'
import { initDatabaseModels } from '../server/initializers/database'
import { optimizeOriginalVideofile } from '../server/lib/transcoding/video-transcoding'
import { VideoModel } from '../server/models/video/video'

registerTSPaths()

run()
.then(() => process.exit(0))
Expand Down Expand Up @@ -39,7 +42,7 @@ async function run () {
currentVideoId = video.id

for (const file of video.VideoFiles) {
currentFilePath = getVideoFilePath(video, file)
currentFilePath = await getVideoFilePathMakeAvailable(video, file)

const [ videoBitrate, fps, dataResolution ] = await Promise.all([
getVideoFileBitrate(currentFilePath),
Expand Down Expand Up @@ -77,6 +80,10 @@ async function run () {
await file.save()
}
}

if (CONFIG.OBJECT_STORAGE.ENABLED === true) {
await processMoveToObjectStorage({ data: { videoUUID: video.uuid } as MoveObjectStoragePayload } as any)
}
}

console.log('Finished optimizing videos')
Expand Down
22 changes: 15 additions & 7 deletions server/controllers/api/videos/upload.ts
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,14 @@ import { deleteResumableUploadMetaFile, getResumableUploadPath } from '@server/h
import { uuidToShort } from '@server/helpers/uuid'
import { createTorrentAndSetInfoHash } from '@server/helpers/webtorrent'
import { getLocalVideoActivityPubUrl } from '@server/lib/activitypub/url'
import { addOptimizeOrMergeAudioJob, buildLocalVideoFromReq, buildVideoThumbnailsFromReq, setVideoTags } from '@server/lib/video'
import {
addMoveToObjectStorageJob,
addOptimizeOrMergeAudioJob,
buildLocalVideoFromReq,
buildNextVideoState,
buildVideoThumbnailsFromReq,
setVideoTags
} from '@server/lib/video'
import { generateWebTorrentVideoFilename, getVideoFilePath } from '@server/lib/video-paths'
import { openapiOperationDoc } from '@server/middlewares/doc'
import { MVideo, MVideoFile, MVideoFullLight } from '@server/types/models'
Expand Down Expand Up @@ -139,10 +146,7 @@ async function addVideo (options: {

const videoData = buildLocalVideoFromReq(videoInfo, videoChannel.id)

videoData.state = CONFIG.TRANSCODING.ENABLED
? VideoState.TO_TRANSCODE
: VideoState.PUBLISHED

videoData.state = buildNextVideoState()
videoData.duration = videoPhysicalFile.duration // duration was added by a previous middleware

const video = new VideoModel(videoData) as MVideoFullLight
Expand Down Expand Up @@ -210,9 +214,13 @@ async function addVideo (options: {

createTorrentFederate(video, videoFile)
.then(() => {
if (video.state !== VideoState.TO_TRANSCODE) return
if (video.state === VideoState.TO_MOVE_TO_EXTERNAL_STORAGE) {
return addMoveToObjectStorageJob(video)
}

return addOptimizeOrMergeAudioJob(videoCreated, videoFile, user)
if (video.state === VideoState.TO_TRANSCODE) {
return addOptimizeOrMergeAudioJob(videoCreated, videoFile, user)
}
})
.catch(err => logger.error('Cannot add optimize/merge audio job for %s.', videoCreated.uuid, { err, ...lTags(videoCreated.uuid) }))

Expand Down
10 changes: 9 additions & 1 deletion server/controllers/download.ts
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,7 @@ import { VideosTorrentCache } from '@server/lib/files-cache/videos-torrent-cache
import { Hooks } from '@server/lib/plugins/hooks'
import { getVideoFilePath } from '@server/lib/video-paths'
import { MStreamingPlaylist, MVideo, MVideoFile, MVideoFullLight } from '@server/types/models'
import { HttpStatusCode, VideoStreamingPlaylistType } from '@shared/models'
import { HttpStatusCode, VideoStorage, VideoStreamingPlaylistType } from '@shared/models'
import { STATIC_DOWNLOAD_PATHS } from '../initializers/constants'
import { asyncMiddleware, videosDownloadValidator } from '../middlewares'

Expand Down Expand Up @@ -81,6 +81,10 @@ async function downloadVideoFile (req: express.Request, res: express.Response) {

if (!checkAllowResult(res, allowParameters, allowedResult)) return

if (videoFile.storage === VideoStorage.OBJECT_STORAGE) {
return res.redirect(videoFile.getObjectStorageUrl())
}

return res.download(getVideoFilePath(video, videoFile), `${video.name}-${videoFile.resolution}p${videoFile.extname}`)
}

Expand All @@ -107,6 +111,10 @@ async function downloadHLSVideoFile (req: express.Request, res: express.Response

if (!checkAllowResult(res, allowParameters, allowedResult)) return

if (videoFile.storage === VideoStorage.OBJECT_STORAGE) {
return res.redirect(videoFile.getObjectStorageUrl())
}

const filename = `${video.name}-${videoFile.resolution}p-${streamingPlaylist.getStringType()}${videoFile.extname}`
return res.download(getVideoFilePath(streamingPlaylist, videoFile), filename)
}
Expand Down
23 changes: 23 additions & 0 deletions server/initializers/checker-after-init.ts
Original file line number Diff line number Diff line change
Expand Up @@ -153,6 +153,29 @@ function checkConfig () {
}
}

// Object storage
if (CONFIG.OBJECT_STORAGE.ENABLED === true) {

if (!CONFIG.OBJECT_STORAGE.VIDEOS.BUCKET_NAME) {
return 'videos_bucket should be set when object storage support is enabled.'
}

if (!CONFIG.OBJECT_STORAGE.STREAMING_PLAYLISTS.BUCKET_NAME) {
return 'streaming_playlists_bucket should be set when object storage support is enabled.'
}

if (
CONFIG.OBJECT_STORAGE.VIDEOS.BUCKET_NAME === CONFIG.OBJECT_STORAGE.STREAMING_PLAYLISTS.BUCKET_NAME &&
CONFIG.OBJECT_STORAGE.VIDEOS.PREFIX === CONFIG.OBJECT_STORAGE.STREAMING_PLAYLISTS.PREFIX
) {
if (CONFIG.OBJECT_STORAGE.VIDEOS.PREFIX === '') {
return 'Object storage bucket prefixes should be set when the same bucket is used for both types of video.'
} else {
return 'Object storage bucket prefixes should be set to different values when the same bucket is used for both types of video.'
}
}
}

return null
}

Expand Down
20 changes: 20 additions & 0 deletions server/initializers/config.ts
Original file line number Diff line number Diff line change
Expand Up @@ -73,6 +73,26 @@ const CONFIG = {
PLUGINS_DIR: buildPath(config.get<string>('storage.plugins')),
CLIENT_OVERRIDES_DIR: buildPath(config.get<string>('storage.client_overrides'))
},
OBJECT_STORAGE: {
ENABLED: config.get<boolean>('object_storage.enabled'),
MAX_UPLOAD_PART: bytes.parse(config.get<string>('object_storage.max_upload_part')),
ENDPOINT: config.get<string>('object_storage.endpoint'),
REGION: config.get<string>('object_storage.region'),
CREDENTIALS: {
ACCESS_KEY_ID: config.get<string>('object_storage.credentials.access_key_id'),
SECRET_ACCESS_KEY: config.get<string>('object_storage.credentials.secret_access_key')
},
VIDEOS: {
BUCKET_NAME: config.get<string>('object_storage.videos.bucket_name'),
PREFIX: config.get<string>('object_storage.videos.prefix'),
BASE_URL: config.get<string>('object_storage.videos.base_url')
},
STREAMING_PLAYLISTS: {
BUCKET_NAME: config.get<string>('object_storage.streaming_playlists.bucket_name'),
PREFIX: config.get<string>('object_storage.streaming_playlists.prefix'),
BASE_URL: config.get<string>('object_storage.streaming_playlists.base_url')
}
},
WEBSERVER: {
SCHEME: config.get<boolean>('webserver.https') === true ? 'https' : 'http',
WS: config.get<boolean>('webserver.https') === true ? 'wss' : 'ws',
Expand Down
14 changes: 9 additions & 5 deletions server/initializers/constants.ts
Original file line number Diff line number Diff line change
Expand Up @@ -24,7 +24,7 @@ import { CONFIG, registerConfigChangedHandler } from './config'

// ---------------------------------------------------------------------------

const LAST_MIGRATION_VERSION = 655
const LAST_MIGRATION_VERSION = 660

// ---------------------------------------------------------------------------

Expand Down Expand Up @@ -147,7 +147,8 @@ const JOB_ATTEMPTS: { [id in JobType]: number } = {
'videos-views': 1,
'activitypub-refresher': 1,
'video-redundancy': 1,
'video-live-ending': 1
'video-live-ending': 1,
'move-to-object-storage': 3
}
// Excluded keys are jobs that can be configured by admins
const JOB_CONCURRENCY: { [id in Exclude<JobType, 'video-transcoding' | 'video-import'>]: number } = {
Expand All @@ -162,7 +163,8 @@ const JOB_CONCURRENCY: { [id in Exclude<JobType, 'video-transcoding' | 'video-im
'videos-views': 1,
'activitypub-refresher': 1,
'video-redundancy': 1,
'video-live-ending': 10
'video-live-ending': 10,
'move-to-object-storage': 1
}
const JOB_TTL: { [id in JobType]: number } = {
'activitypub-http-broadcast': 60000 * 10, // 10 minutes
Expand All @@ -178,7 +180,8 @@ const JOB_TTL: { [id in JobType]: number } = {
'videos-views': undefined, // Unlimited
'activitypub-refresher': 60000 * 10, // 10 minutes
'video-redundancy': 1000 * 3600 * 3, // 3 hours
'video-live-ending': 1000 * 60 * 10 // 10 minutes
'video-live-ending': 1000 * 60 * 10, // 10 minutes
'move-to-object-storage': 1000 * 60 * 60 * 3 // 3 hours
}
const REPEAT_JOBS: { [ id: string ]: EveryRepeatOptions | CronRepeatOptions } = {
'videos-views': {
Expand Down Expand Up @@ -412,7 +415,8 @@ const VIDEO_STATES: { [ id in VideoState ]: string } = {
[VideoState.TO_TRANSCODE]: 'To transcode',
[VideoState.TO_IMPORT]: 'To import',
[VideoState.WAITING_FOR_LIVE]: 'Waiting for livestream',
[VideoState.LIVE_ENDED]: 'Livestream ended'
[VideoState.LIVE_ENDED]: 'Livestream ended',
[VideoState.TO_MOVE_TO_EXTERNAL_STORAGE]: 'To move to an external storage'
}

const VIDEO_IMPORT_STATES: { [ id in VideoImportState ]: string } = {
Expand Down
4 changes: 3 additions & 1 deletion server/initializers/database.ts
Original file line number Diff line number Diff line change
Expand Up @@ -45,6 +45,7 @@ import { VideoTagModel } from '../models/video/video-tag'
import { VideoViewModel } from '../models/video/video-view'
import { CONFIG } from './config'
import { ActorCustomPageModel } from '@server/models/account/actor-custom-page'
import { VideoJobInfoModel } from '@server/models/video/video-job-info'

require('pg').defaults.parseInt8 = true // Avoid BIGINT to be converted to string

Expand Down Expand Up @@ -143,7 +144,8 @@ async function initDatabaseModels (silent: boolean) {
TrackerModel,
VideoTrackerModel,
PluginModel,
ActorCustomPageModel
ActorCustomPageModel,
VideoJobInfoModel
])

// Check extensions exist in the database
Expand Down
58 changes: 58 additions & 0 deletions server/initializers/migrations/0660-object-storage.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,58 @@
import * as Sequelize from 'sequelize'
import { VideoStorage } from '@shared/models'

async function up (utils: {
transaction: Sequelize.Transaction
queryInterface: Sequelize.QueryInterface
sequelize: Sequelize.Sequelize
db: any
}): Promise<void> {
{
const query = `
CREATE TABLE IF NOT EXISTS "videoJobInfo" (
"id" serial,
"pendingMove" INTEGER NOT NULL,
"pendingTranscoding" INTEGER NOT NULL,
"videoId" serial UNIQUE NOT NULL REFERENCES "video" ("id") ON DELETE CASCADE ON UPDATE CASCADE,
"createdAt" timestamp WITH time zone NOT NULL,
"updatedAt" timestamp WITH time zone NOT NULL,
PRIMARY KEY ("id")
);
`

await utils.sequelize.query(query)
}

{
await utils.queryInterface.addColumn('videoFile', 'storage', { type: Sequelize.INTEGER, allowNull: true })
}
{
await utils.sequelize.query(
`UPDATE "videoFile" SET "storage" = ${VideoStorage.LOCAL}`
)
}
{
await utils.queryInterface.changeColumn('videoFile', 'storage', { type: Sequelize.INTEGER, allowNull: false })
}

{
await utils.queryInterface.addColumn('videoStreamingPlaylist', 'storage', { type: Sequelize.INTEGER, allowNull: true })
}
{
await utils.sequelize.query(
`UPDATE "videoStreamingPlaylist" SET "storage" = ${VideoStorage.LOCAL}`
)
}
{
await utils.queryInterface.changeColumn('videoStreamingPlaylist', 'storage', { type: Sequelize.INTEGER, allowNull: false })
}
}

function down (options) {
throw new Error('Not implemented.')
}

export {
up,
down
}
Loading