import { join } from 'path'
import { createServer as createServerTLS, Server as ServerTLS } from 'tls'
import {
- computeLowerResolutionsToTranscode,
+ computeResolutionsToTranscode,
ffprobePromise,
getLiveSegmentTime,
getVideoStreamBitrate,
import { JobQueue } from '../job-queue'
import { generateHLSMasterPlaylistFilename, generateHlsSha256SegmentsFilename, getLiveReplayBaseDirectory } from '../paths'
import { PeerTubeSocket } from '../peertube-socket'
+import { Hooks } from '../plugins/hooks'
import { LiveQuotaStore } from './live-quota-store'
import { cleanupPermanentLive } from './live-utils'
import { MuxingSession } from './shared'
inputUrl, Date.now() - now, bitrate, fps, resolution, lTags(sessionId, video.uuid)
)
- const allResolutions = this.buildAllResolutionsToTranscode(resolution)
+ const allResolutions = await Hooks.wrapObject(
+ this.buildAllResolutionsToTranscode(resolution),
+ 'filter:transcoding.auto.resolutions-to-transcode.result',
+ { video }
+ )
logger.info(
'Will mux/transcode live video of original resolution %d.', resolution,
const videoId = live.videoId
try {
- const video = await VideoModel.loadAndPopulateAccountAndServerAndTags(videoId)
+ const video = await VideoModel.loadFull(videoId)
logger.info('Will publish and federate live %s.', video.url, localLTags)
const { videoId, liveSession: liveSessionArg, cleanupNow = false } = options
try {
- const fullVideo = await VideoModel.loadAndPopulateAccountAndServerAndTags(videoId)
+ const fullVideo = await VideoModel.loadFull(videoId)
if (!fullVideo) return
const live = await VideoLiveModel.loadByVideoId(fullVideo.id)
- const liveSession = liveSessionArg ?? await VideoLiveSessionModel.findCurrentSessionOf(fullVideo.id)
+ const liveSession = liveSessionArg ?? await VideoLiveSessionModel.findLatestSessionOf(fullVideo.id)
// On server restart during a live
if (!liveSession.endDate) {
await liveSession.save()
}
- JobQueue.Instance.createJob({
+ JobQueue.Instance.createJobAsync({
type: 'video-live-ending',
payload: {
videoId: fullVideo.id,
streamingPlaylistId: fullVideo.getHLSPlaylist()?.id,
publishedAt: fullVideo.publishedAt.toISOString()
- }
- }, { delay: cleanupNow ? 0 : VIDEO_LIVE.CLEANUP_DELAY })
+ },
+
+ delay: cleanupNow
+ ? 0
+ : VIDEO_LIVE.CLEANUP_DELAY
+ })
fullVideo.state = live.permanentLive
? VideoState.WAITING_FOR_LIVE
}
private buildAllResolutionsToTranscode (originResolution: number) {
+ const includeInput = CONFIG.LIVE.TRANSCODING.ALWAYS_TRANSCODE_ORIGINAL_RESOLUTION
+
const resolutionsEnabled = CONFIG.LIVE.TRANSCODING.ENABLED
- ? computeLowerResolutionsToTranscode(originResolution, 'live')
+ ? computeResolutionsToTranscode({ input: originResolution, type: 'live', includeInput, strictLower: false })
: []
- return resolutionsEnabled.concat([ originResolution ])
+ if (resolutionsEnabled.length === 0) {
+ return [ originResolution ]
+ }
+
+ return resolutionsEnabled
}
private async createLivePlaylist (video: MVideo, allResolutions: number[]): Promise<MStreamingPlaylistVideo> {
private saveStartingSession (videoLive: MVideoLiveVideo) {
const liveSession = new VideoLiveSessionModel({
startDate: new Date(),
- liveVideoId: videoLive.videoId
+ liveVideoId: videoLive.videoId,
+ saveReplay: videoLive.saveReplay,
+ endingProcessed: false
})
return liveSession.save()
private async saveEndingSession (videoId: number, error: LiveVideoError | null) {
const liveSession = await VideoLiveSessionModel.findCurrentSessionOf(videoId)
+ if (!liveSession) return
+
liveSession.endDate = new Date()
liveSession.error = error