2022-06-23 03:29:43 -05:00
|
|
|
/* eslint-disable @typescript-eslint/no-unused-expressions,@typescript-eslint/require-await */
|
|
|
|
|
2022-08-17 08:44:32 -05:00
|
|
|
import { expect } from 'chai'
|
2022-06-23 03:29:43 -05:00
|
|
|
import { wait } from '@shared/core-utils'
|
|
|
|
import { HttpStatusCode, LiveVideoCreate, VideoPrivacy } from '@shared/models'
|
|
|
|
import {
|
|
|
|
cleanupTests,
|
|
|
|
createSingleServer,
|
|
|
|
makeRawRequest,
|
|
|
|
PeerTubeServer,
|
|
|
|
setAccessTokensToServers,
|
|
|
|
setDefaultVideoChannel,
|
|
|
|
stopFfmpeg,
|
|
|
|
waitJobs
|
|
|
|
} from '@shared/server-commands'
|
|
|
|
|
|
|
|
describe('Fast restream in live', function () {
|
|
|
|
let server: PeerTubeServer
|
|
|
|
|
|
|
|
async function createLiveWrapper (options: { permanent: boolean, replay: boolean }) {
|
|
|
|
const attributes: LiveVideoCreate = {
|
|
|
|
channelId: server.store.channel.id,
|
|
|
|
privacy: VideoPrivacy.PUBLIC,
|
|
|
|
name: 'my super live',
|
|
|
|
saveReplay: options.replay,
|
|
|
|
permanentLive: options.permanent
|
|
|
|
}
|
|
|
|
|
|
|
|
const { uuid } = await server.live.create({ fields: attributes })
|
|
|
|
return uuid
|
|
|
|
}
|
|
|
|
|
|
|
|
async function fastRestreamWrapper ({ replay }: { replay: boolean }) {
|
|
|
|
const liveVideoUUID = await createLiveWrapper({ permanent: true, replay })
|
|
|
|
await waitJobs([ server ])
|
|
|
|
|
|
|
|
const rtmpOptions = {
|
|
|
|
videoId: liveVideoUUID,
|
|
|
|
copyCodecs: true,
|
|
|
|
fixtureName: 'video_short.mp4'
|
|
|
|
}
|
|
|
|
|
|
|
|
// Streaming session #1
|
|
|
|
let ffmpegCommand = await server.live.sendRTMPStreamInVideo(rtmpOptions)
|
|
|
|
await server.live.waitUntilPublished({ videoId: liveVideoUUID })
|
2022-10-11 09:00:11 -05:00
|
|
|
|
|
|
|
const video = await server.videos.get({ id: liveVideoUUID })
|
|
|
|
const session1PlaylistId = video.streamingPlaylists[0].id
|
|
|
|
|
2022-06-23 03:29:43 -05:00
|
|
|
await stopFfmpeg(ffmpegCommand)
|
|
|
|
await server.live.waitUntilWaiting({ videoId: liveVideoUUID })
|
|
|
|
|
|
|
|
// Streaming session #2
|
|
|
|
ffmpegCommand = await server.live.sendRTMPStreamInVideo(rtmpOptions)
|
2022-10-11 09:00:11 -05:00
|
|
|
|
|
|
|
let hasNewPlaylist = false
|
|
|
|
do {
|
|
|
|
const video = await server.videos.get({ id: liveVideoUUID })
|
|
|
|
hasNewPlaylist = video.streamingPlaylists.length === 1 && video.streamingPlaylists[0].id !== session1PlaylistId
|
|
|
|
|
|
|
|
await wait(100)
|
|
|
|
} while (!hasNewPlaylist)
|
|
|
|
|
|
|
|
await server.live.waitUntilSegmentGeneration({
|
|
|
|
server,
|
|
|
|
videoUUID: liveVideoUUID,
|
|
|
|
segment: 1,
|
|
|
|
playlistNumber: 0,
|
|
|
|
objectStorage: false
|
|
|
|
})
|
2022-06-23 03:29:43 -05:00
|
|
|
|
|
|
|
return { ffmpegCommand, liveVideoUUID }
|
|
|
|
}
|
|
|
|
|
|
|
|
async function ensureLastLiveWorks (liveId: string) {
|
2022-06-24 02:34:56 -05:00
|
|
|
// Equivalent to PEERTUBE_TEST_CONSTANTS_VIDEO_LIVE_CLEANUP_DELAY
|
2022-06-23 03:29:43 -05:00
|
|
|
for (let i = 0; i < 100; i++) {
|
|
|
|
const video = await server.videos.get({ id: liveId })
|
|
|
|
expect(video.streamingPlaylists).to.have.lengthOf(1)
|
|
|
|
|
2022-10-04 03:03:17 -05:00
|
|
|
await server.live.getSegmentFile({ videoUUID: liveId, segment: 0, playlistNumber: 0 })
|
2022-06-23 03:29:43 -05:00
|
|
|
await makeRawRequest(video.streamingPlaylists[0].playlistUrl, HttpStatusCode.OK_200)
|
2022-06-24 02:34:56 -05:00
|
|
|
await makeRawRequest(video.streamingPlaylists[0].segmentsSha256Url, HttpStatusCode.OK_200)
|
2022-06-23 03:29:43 -05:00
|
|
|
|
|
|
|
await wait(100)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
async function runTest (replay: boolean) {
|
|
|
|
const { ffmpegCommand, liveVideoUUID } = await fastRestreamWrapper({ replay })
|
|
|
|
|
|
|
|
await ensureLastLiveWorks(liveVideoUUID)
|
|
|
|
|
|
|
|
await stopFfmpeg(ffmpegCommand)
|
|
|
|
await server.live.waitUntilWaiting({ videoId: liveVideoUUID })
|
|
|
|
|
|
|
|
// Wait for replays
|
|
|
|
await waitJobs([ server ])
|
|
|
|
|
|
|
|
const { total, data: sessions } = await server.live.listSessions({ videoId: liveVideoUUID })
|
|
|
|
|
|
|
|
expect(total).to.equal(2)
|
|
|
|
expect(sessions).to.have.lengthOf(2)
|
|
|
|
|
|
|
|
for (const session of sessions) {
|
|
|
|
expect(session.error).to.be.null
|
|
|
|
|
|
|
|
if (replay) {
|
|
|
|
expect(session.replayVideo).to.exist
|
|
|
|
|
|
|
|
await server.videos.get({ id: session.replayVideo.uuid })
|
|
|
|
} else {
|
|
|
|
expect(session.replayVideo).to.not.exist
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
before(async function () {
|
|
|
|
this.timeout(120000)
|
|
|
|
|
2022-06-24 03:13:09 -05:00
|
|
|
const env = { PEERTUBE_TEST_CONSTANTS_VIDEO_LIVE_CLEANUP_DELAY: '10000' }
|
2022-06-23 03:29:43 -05:00
|
|
|
server = await createSingleServer(1, {}, { env })
|
|
|
|
|
|
|
|
// Get the access tokens
|
|
|
|
await setAccessTokensToServers([ server ])
|
|
|
|
await setDefaultVideoChannel([ server ])
|
|
|
|
|
|
|
|
await server.config.enableMinimumTranscoding(false, true)
|
|
|
|
await server.config.enableLive({ allowReplay: true, transcoding: true, resolutions: 'min' })
|
|
|
|
})
|
|
|
|
|
|
|
|
it('Should correctly fast reastream in a permanent live with and without save replay', async function () {
|
|
|
|
this.timeout(240000)
|
|
|
|
|
|
|
|
// A test can take a long time, so prefer to run them in parallel
|
|
|
|
await Promise.all([
|
|
|
|
runTest(true),
|
|
|
|
runTest(false)
|
|
|
|
])
|
|
|
|
})
|
|
|
|
|
|
|
|
after(async function () {
|
|
|
|
await cleanupTests([ server ])
|
|
|
|
})
|
|
|
|
})
|