2021-10-13 02:44:34 -04:00
import config from 'config'
2020-01-31 10:56:52 -05:00
import { URL } from 'url'
2022-02-11 04:51:33 -05:00
import { getFFmpegVersion } from '@server/helpers/ffmpeg'
2022-08-17 09:36:03 -04:00
import { uniqify } from '@shared/core-utils'
2021-03-11 03:51:08 -05:00
import { VideoRedundancyConfigFilter } from '@shared/models/redundancy/video-redundancy-config-filter.type'
2018-10-01 07:29:38 -04:00
import { RecentlyAddedStrategy } from '../../shared/models/redundancy'
2022-07-06 09:44:14 -04:00
import { isProdInstance , parseSemVersion } from '../helpers/core-utils'
2018-09-24 07:07:33 -04:00
import { isArray } from '../helpers/custom-validators/misc'
2021-03-11 03:51:08 -05:00
import { logger } from '../helpers/logger'
import { ApplicationModel , getServerActor } from '../models/application/application'
import { OAuthClientModel } from '../models/oauth/oauth-client'
2021-10-13 02:44:34 -04:00
import { UserModel } from '../models/user/user'
2021-03-11 03:51:08 -05:00
import { CONFIG , isEmailEnabled } from './config'
2019-04-11 05:33:44 -04:00
import { WEBSERVER } from './constants'
2018-09-24 07:07:33 -04:00
async function checkActivityPubUrls ( ) {
const actor = await getServerActor ( )
2020-01-31 10:56:52 -05:00
const parsed = new URL ( actor . url )
2019-04-11 05:33:44 -04:00
if ( WEBSERVER . HOST !== parsed . host ) {
2021-10-13 02:44:34 -04:00
const NODE_ENV = config . util . getEnv ( 'NODE_ENV' )
const NODE_CONFIG_DIR = config . util . getEnv ( 'NODE_CONFIG_DIR' )
2018-09-24 07:07:33 -04:00
logger . warn (
'It seems PeerTube was started (and created some data) with another domain name. ' +
'This means you will not be able to federate! ' +
'Please use %s %s npm run update-host to fix this.' ,
NODE_CONFIG_DIR ? ` NODE_CONFIG_DIR= ${ NODE_CONFIG_DIR } ` : '' ,
NODE_ENV ? ` NODE_ENV= ${ NODE_ENV } ` : ''
)
}
}
2022-02-11 04:51:33 -05:00
// Some checks on configuration files or throw if there is an error
2018-09-24 07:07:33 -04:00
function checkConfig ( ) {
2019-01-10 05:12:41 -05:00
2022-07-06 09:44:14 -04:00
const configFiles = config . util . getConfigSources ( ) . map ( s = > s . name ) . join ( ' -> ' )
logger . info ( 'Using following configuration file hierarchy: %s.' , configFiles )
2019-02-21 10:27:32 -05:00
// Moved configuration keys
2021-10-13 02:44:34 -04:00
if ( config . has ( 'services.csp-logger' ) ) {
2019-02-21 10:27:32 -05:00
logger . warn ( 'services.csp-logger configuration has been renamed to csp.report_uri. Please update your configuration file.' )
}
2022-10-10 05:12:23 -04:00
checkSecretsConfig ( )
2022-02-11 04:51:33 -05:00
checkEmailConfig ( )
checkNSFWPolicyConfig ( )
checkLocalRedundancyConfig ( )
checkRemoteRedundancyConfig ( )
checkStorageConfig ( )
checkTranscodingConfig ( )
2022-08-10 03:53:39 -04:00
checkImportConfig ( )
2022-02-11 04:51:33 -05:00
checkBroadcastMessageConfig ( )
checkSearchConfig ( )
checkLiveConfig ( )
checkObjectStorageConfig ( )
2022-03-22 11:58:49 -04:00
checkVideoStudioConfig ( )
2022-02-11 04:51:33 -05:00
}
// We get db by param to not import it in this file (import orders)
async function clientsExist ( ) {
const totalClients = await OAuthClientModel . countTotal ( )
return totalClients !== 0
}
// We get db by param to not import it in this file (import orders)
async function usersExist ( ) {
const totalUsers = await UserModel . countTotal ( )
return totalUsers !== 0
}
// We get db by param to not import it in this file (import orders)
async function applicationExist ( ) {
const totalApplication = await ApplicationModel . countTotal ( )
return totalApplication !== 0
}
async function checkFFmpegVersion ( ) {
const version = await getFFmpegVersion ( )
2022-07-18 08:53:20 -04:00
const { major , minor , patch } = parseSemVersion ( version )
2022-02-11 04:51:33 -05:00
if ( major < 4 || ( major === 4 && minor < 1 ) ) {
2022-07-18 08:53:20 -04:00
logger . warn ( 'Your ffmpeg version (%s) is outdated. PeerTube supports ffmpeg >= 4.1. Please upgrade ffmpeg.' , version )
}
if ( major === 4 && minor === 4 && patch === 0 ) {
logger . warn ( 'There is a bug in ffmpeg 4.4.0 with HLS videos. Please upgrade ffmpeg.' )
2022-02-11 04:51:33 -05:00
}
}
// ---------------------------------------------------------------------------
export {
checkConfig ,
clientsExist ,
checkFFmpegVersion ,
usersExist ,
applicationExist ,
checkActivityPubUrls
}
// ---------------------------------------------------------------------------
2022-10-10 05:12:23 -04:00
function checkSecretsConfig ( ) {
if ( ! CONFIG . SECRETS . PEERTUBE ) {
throw new Error ( 'secrets.peertube is missing in config. Generate one using `openssl rand -hex 32`' )
}
}
2022-02-11 04:51:33 -05:00
function checkEmailConfig ( ) {
2020-02-17 04:27:00 -05:00
if ( ! isEmailEnabled ( ) ) {
2019-01-10 05:12:41 -05:00
if ( CONFIG . SIGNUP . ENABLED && CONFIG . SIGNUP . REQUIRES_EMAIL_VERIFICATION ) {
2022-02-11 04:51:33 -05:00
throw new Error ( 'Emailer is disabled but you require signup email verification.' )
2019-01-10 05:12:41 -05:00
}
2023-01-19 03:27:16 -05:00
if ( CONFIG . SIGNUP . ENABLED && CONFIG . SIGNUP . REQUIRES_APPROVAL ) {
// eslint-disable-next-line max-len
logger . warn ( 'Emailer is disabled but signup approval is enabled: PeerTube will not be able to send an email to the user upon acceptance/rejection of the registration request' )
}
2019-01-10 05:12:41 -05:00
if ( CONFIG . CONTACT_FORM . ENABLED ) {
logger . warn ( 'Emailer is disabled so the contact form will not work.' )
}
}
2022-02-11 04:51:33 -05:00
}
2018-09-24 07:07:33 -04:00
2022-02-11 04:51:33 -05:00
function checkNSFWPolicyConfig ( ) {
2019-01-10 05:12:41 -05:00
const defaultNSFWPolicy = CONFIG . INSTANCE . DEFAULT_NSFW_POLICY
2022-02-11 04:51:33 -05:00
const available = [ 'do_not_list' , 'blur' , 'display' ]
if ( available . includes ( defaultNSFWPolicy ) === false ) {
throw new Error ( 'NSFW policy setting should be ' + available . join ( ' or ' ) + ' instead of ' + defaultNSFWPolicy )
2018-09-24 07:07:33 -04:00
}
2022-02-11 04:51:33 -05:00
}
2018-09-24 07:07:33 -04:00
2022-02-11 04:51:33 -05:00
function checkLocalRedundancyConfig ( ) {
2018-09-24 07:07:33 -04:00
const redundancyVideos = CONFIG . REDUNDANCY . VIDEOS . STRATEGIES
2022-02-11 04:51:33 -05:00
2018-09-24 07:07:33 -04:00
if ( isArray ( redundancyVideos ) ) {
const available = [ 'most-views' , 'trending' , 'recently-added' ]
2022-02-11 04:51:33 -05:00
2018-09-24 07:07:33 -04:00
for ( const r of redundancyVideos ) {
2020-02-28 10:03:39 -05:00
if ( available . includes ( r . strategy ) === false ) {
2022-02-11 04:51:33 -05:00
throw new Error ( 'Videos redundancy should have ' + available . join ( ' or ' ) + ' strategy instead of ' + r . strategy )
2018-09-24 07:07:33 -04:00
}
// Lifetime should not be < 10 hours
2022-07-06 09:44:14 -04:00
if ( isProdInstance ( ) && r . minLifetime < 1000 * 3600 * 10 ) {
2022-02-11 04:51:33 -05:00
throw new Error ( 'Video redundancy minimum lifetime should be >= 10 hours for strategy ' + r . strategy )
2018-09-24 07:07:33 -04:00
}
}
2022-08-17 09:36:03 -04:00
const filtered = uniqify ( redundancyVideos . map ( r = > r . strategy ) )
2018-09-24 07:07:33 -04:00
if ( filtered . length !== redundancyVideos . length ) {
2022-02-11 04:51:33 -05:00
throw new Error ( 'Redundancy video entries should have unique strategies' )
2018-09-24 07:07:33 -04:00
}
const recentlyAddedStrategy = redundancyVideos . find ( r = > r . strategy === 'recently-added' ) as RecentlyAddedStrategy
if ( recentlyAddedStrategy && isNaN ( recentlyAddedStrategy . minViews ) ) {
2022-02-11 04:51:33 -05:00
throw new Error ( 'Min views in recently added strategy is not a number' )
2018-09-24 07:07:33 -04:00
}
2019-03-19 11:33:40 -04:00
} else {
2022-02-11 04:51:33 -05:00
throw new Error ( 'Videos redundancy should be an array (you must uncomment lines containing - too)' )
2018-09-24 07:07:33 -04:00
}
2022-02-11 04:51:33 -05:00
}
2018-09-24 07:07:33 -04:00
2022-02-11 04:51:33 -05:00
function checkRemoteRedundancyConfig ( ) {
2020-04-07 09:27:41 -04:00
const acceptFrom = CONFIG . REMOTE_REDUNDANCY . VIDEOS . ACCEPT_FROM
const acceptFromValues = new Set < VideoRedundancyConfigFilter > ( [ 'nobody' , 'anybody' , 'followings' ] )
2022-02-11 04:51:33 -05:00
2020-04-07 09:27:41 -04:00
if ( acceptFromValues . has ( acceptFrom ) === false ) {
2022-02-11 04:51:33 -05:00
throw new Error ( 'remote_redundancy.videos.accept_from has an incorrect value' )
2020-04-07 09:27:41 -04:00
}
2022-02-11 04:51:33 -05:00
}
2020-04-07 09:27:41 -04:00
2022-02-11 04:51:33 -05:00
function checkStorageConfig ( ) {
2019-01-10 05:12:41 -05:00
// Check storage directory locations
2018-09-24 07:07:33 -04:00
if ( isProdInstance ( ) ) {
2022-12-30 04:12:20 -05:00
const configStorage = config . get < { [ name : string ] : string } > ( 'storage' )
2018-09-24 07:07:33 -04:00
for ( const key of Object . keys ( configStorage ) ) {
if ( configStorage [ key ] . startsWith ( 'storage/' ) ) {
logger . warn (
'Directory of %s should not be in the production directory of PeerTube. Please check your production configuration file.' ,
key
)
}
}
}
2020-05-28 05:15:38 -04:00
if ( CONFIG . STORAGE . VIDEOS_DIR === CONFIG . STORAGE . REDUNDANCY_DIR ) {
logger . warn ( 'Redundancy directory should be different than the videos folder.' )
}
2022-02-11 04:51:33 -05:00
}
2020-05-28 05:15:38 -04:00
2022-02-11 04:51:33 -05:00
function checkTranscodingConfig ( ) {
2019-11-15 09:06:03 -05:00
if ( CONFIG . TRANSCODING . ENABLED ) {
if ( CONFIG . TRANSCODING . WEBTORRENT . ENABLED === false && CONFIG . TRANSCODING . HLS . ENABLED === false ) {
2022-02-11 04:51:33 -05:00
throw new Error ( 'You need to enable at least WebTorrent transcoding or HLS transcoding.' )
2019-11-15 09:06:03 -05:00
}
2021-02-08 04:51:10 -05:00
if ( CONFIG . TRANSCODING . CONCURRENCY <= 0 ) {
2022-02-11 04:51:33 -05:00
throw new Error ( 'Transcoding concurrency should be > 0' )
2021-02-08 04:51:10 -05:00
}
}
if ( CONFIG . IMPORT . VIDEOS . HTTP . ENABLED || CONFIG . IMPORT . VIDEOS . TORRENT . ENABLED ) {
if ( CONFIG . IMPORT . VIDEOS . CONCURRENCY <= 0 ) {
2022-02-11 04:51:33 -05:00
throw new Error ( 'Video import concurrency should be > 0' )
2021-02-08 04:51:10 -05:00
}
2019-11-15 09:06:03 -05:00
}
2022-02-11 04:51:33 -05:00
}
2019-11-15 09:06:03 -05:00
2022-08-10 03:53:39 -04:00
function checkImportConfig ( ) {
if ( CONFIG . IMPORT . VIDEO_CHANNEL_SYNCHRONIZATION . ENABLED && ! CONFIG . IMPORT . VIDEOS . HTTP ) {
throw new Error ( 'You need to enable HTTP import to allow synchronization' )
}
}
2022-02-11 04:51:33 -05:00
function checkBroadcastMessageConfig ( ) {
2020-05-28 05:15:38 -04:00
if ( CONFIG . BROADCAST_MESSAGE . ENABLED ) {
const currentLevel = CONFIG . BROADCAST_MESSAGE . LEVEL
2021-05-25 04:08:29 -04:00
const available = [ 'info' , 'warning' , 'error' ]
2020-05-28 05:15:38 -04:00
if ( available . includes ( currentLevel ) === false ) {
2022-02-11 04:51:33 -05:00
throw new Error ( 'Broadcast message level should be ' + available . join ( ' or ' ) + ' instead of ' + currentLevel )
2020-05-28 05:15:38 -04:00
}
2019-12-10 08:31:08 -05:00
}
2022-02-11 04:51:33 -05:00
}
2019-12-10 08:31:08 -05:00
2022-02-11 04:51:33 -05:00
function checkSearchConfig ( ) {
2020-05-29 10:16:24 -04:00
if ( CONFIG . SEARCH . SEARCH_INDEX . ENABLED === true ) {
if ( CONFIG . SEARCH . REMOTE_URI . USERS === false ) {
2022-02-11 04:51:33 -05:00
throw new Error ( 'You cannot enable search index without enabling remote URI search for users.' )
2020-05-29 10:16:24 -04:00
}
}
2022-02-11 04:51:33 -05:00
}
2020-05-29 10:16:24 -04:00
2022-02-11 04:51:33 -05:00
function checkLiveConfig ( ) {
2020-09-25 10:19:35 -04:00
if ( CONFIG . LIVE . ENABLED === true ) {
if ( CONFIG . LIVE . ALLOW_REPLAY === true && CONFIG . TRANSCODING . ENABLED === false ) {
2022-02-11 04:51:33 -05:00
throw new Error ( 'Live allow replay cannot be enabled if transcoding is not enabled.' )
2020-09-25 10:19:35 -04:00
}
2021-11-05 06:36:03 -04:00
if ( CONFIG . LIVE . RTMP . ENABLED === false && CONFIG . LIVE . RTMPS . ENABLED === false ) {
2022-02-11 04:51:33 -05:00
throw new Error ( 'You must enable at least RTMP or RTMPS' )
2021-11-05 06:36:03 -04:00
}
if ( CONFIG . LIVE . RTMPS . ENABLED ) {
if ( ! CONFIG . LIVE . RTMPS . KEY_FILE ) {
2022-02-11 04:51:33 -05:00
throw new Error ( 'You must specify a key file to enabled RTMPS' )
2021-11-05 06:36:03 -04:00
}
if ( ! CONFIG . LIVE . RTMPS . CERT_FILE ) {
2022-02-11 04:51:33 -05:00
throw new Error ( 'You must specify a cert file to enable RTMPS' )
2021-11-05 06:36:03 -04:00
}
}
2020-09-25 10:19:35 -04:00
}
2022-02-11 04:51:33 -05:00
}
2020-09-25 10:19:35 -04:00
2022-02-11 04:51:33 -05:00
function checkObjectStorageConfig ( ) {
2021-08-17 02:26:20 -04:00
if ( CONFIG . OBJECT_STORAGE . ENABLED === true ) {
if ( ! CONFIG . OBJECT_STORAGE . VIDEOS . BUCKET_NAME ) {
2022-02-11 04:51:33 -05:00
throw new Error ( 'videos_bucket should be set when object storage support is enabled.' )
2021-08-17 02:26:20 -04:00
}
if ( ! CONFIG . OBJECT_STORAGE . STREAMING_PLAYLISTS . BUCKET_NAME ) {
2022-02-11 04:51:33 -05:00
throw new Error ( 'streaming_playlists_bucket should be set when object storage support is enabled.' )
2021-08-17 02:26:20 -04:00
}
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 === '' ) {
2022-02-11 04:51:33 -05:00
throw new Error ( 'Object storage bucket prefixes should be set when the same bucket is used for both types of video.' )
2021-08-17 02:26:20 -04:00
}
2022-02-11 04:51:33 -05:00
throw new Error (
'Object storage bucket prefixes should be set to different values when the same bucket is used for both types of video.'
)
2021-08-17 02:26:20 -04:00
}
}
2018-09-24 07:07:33 -04:00
}
2022-03-22 11:58:49 -04:00
function checkVideoStudioConfig ( ) {
if ( CONFIG . VIDEO_STUDIO . ENABLED === true && CONFIG . TRANSCODING . ENABLED === false ) {
throw new Error ( 'Video studio cannot be enabled if transcoding is disabled' )
2021-03-11 03:51:08 -05:00
}
}