2023-04-21 09:00:01 -04:00
|
|
|
/* eslint-disable @typescript-eslint/no-unused-expressions,@typescript-eslint/require-await */
|
2023-05-19 07:33:27 -04:00
|
|
|
import { expect } from 'chai'
|
2023-05-04 09:29:34 -04:00
|
|
|
import {
|
|
|
|
checkPeerTubeRunnerCacheIsEmpty,
|
|
|
|
expectStartWith,
|
|
|
|
PeerTubeRunnerProcess,
|
|
|
|
SQLCommand,
|
|
|
|
testLiveVideoResolutions
|
|
|
|
} from '@server/tests/shared'
|
2023-04-21 09:00:01 -04:00
|
|
|
import { areMockObjectStorageTestsDisabled, wait } from '@shared/core-utils'
|
|
|
|
import { HttpStatusCode, VideoPrivacy } from '@shared/models'
|
|
|
|
import {
|
|
|
|
cleanupTests,
|
|
|
|
createMultipleServers,
|
|
|
|
doubleFollow,
|
|
|
|
findExternalSavedVideo,
|
|
|
|
makeRawRequest,
|
|
|
|
ObjectStorageCommand,
|
|
|
|
PeerTubeServer,
|
|
|
|
setAccessTokensToServers,
|
|
|
|
setDefaultVideoChannel,
|
|
|
|
stopFfmpeg,
|
|
|
|
waitJobs,
|
|
|
|
waitUntilLivePublishedOnAllServers,
|
|
|
|
waitUntilLiveWaitingOnAllServers
|
|
|
|
} from '@shared/server-commands'
|
|
|
|
|
|
|
|
describe('Test Live transcoding in peertube-runner program', function () {
|
|
|
|
let servers: PeerTubeServer[] = []
|
|
|
|
let peertubeRunner: PeerTubeRunnerProcess
|
|
|
|
let sqlCommandServer1: SQLCommand
|
|
|
|
|
|
|
|
function runSuite (options: {
|
2023-05-23 04:49:45 -04:00
|
|
|
objectStorage?: ObjectStorageCommand
|
|
|
|
} = {}) {
|
2023-04-21 09:00:01 -04:00
|
|
|
const { objectStorage } = options
|
|
|
|
|
|
|
|
it('Should enable transcoding without additional resolutions', async function () {
|
|
|
|
this.timeout(120000)
|
|
|
|
|
|
|
|
const { video } = await servers[0].live.quickCreate({ permanentLive: true, saveReplay: false, privacy: VideoPrivacy.PUBLIC })
|
|
|
|
|
|
|
|
const ffmpegCommand = await servers[0].live.sendRTMPStreamInVideo({ videoId: video.uuid })
|
|
|
|
await waitUntilLivePublishedOnAllServers(servers, video.uuid)
|
|
|
|
await waitJobs(servers)
|
|
|
|
|
|
|
|
await testLiveVideoResolutions({
|
|
|
|
originServer: servers[0],
|
|
|
|
sqlCommand: sqlCommandServer1,
|
|
|
|
servers,
|
|
|
|
liveVideoId: video.uuid,
|
|
|
|
resolutions: [ 720, 480, 360, 240, 144 ],
|
|
|
|
objectStorage,
|
|
|
|
transcoded: true
|
|
|
|
})
|
|
|
|
|
|
|
|
await stopFfmpeg(ffmpegCommand)
|
|
|
|
|
|
|
|
await waitUntilLiveWaitingOnAllServers(servers, video.uuid)
|
|
|
|
await servers[0].videos.remove({ id: video.id })
|
|
|
|
})
|
|
|
|
|
|
|
|
it('Should transcode audio only RTMP stream', async function () {
|
|
|
|
this.timeout(120000)
|
|
|
|
|
|
|
|
const { video } = await servers[0].live.quickCreate({ permanentLive: true, saveReplay: false, privacy: VideoPrivacy.UNLISTED })
|
|
|
|
|
|
|
|
const ffmpegCommand = await servers[0].live.sendRTMPStreamInVideo({ videoId: video.uuid, fixtureName: 'video_short_no_audio.mp4' })
|
|
|
|
await waitUntilLivePublishedOnAllServers(servers, video.uuid)
|
|
|
|
await waitJobs(servers)
|
|
|
|
|
|
|
|
await stopFfmpeg(ffmpegCommand)
|
|
|
|
|
|
|
|
await waitUntilLiveWaitingOnAllServers(servers, video.uuid)
|
|
|
|
await servers[0].videos.remove({ id: video.id })
|
|
|
|
})
|
|
|
|
|
|
|
|
it('Should save a replay', async function () {
|
2023-05-22 02:10:52 -04:00
|
|
|
this.timeout(240000)
|
2023-04-21 09:00:01 -04:00
|
|
|
|
|
|
|
const { video } = await servers[0].live.quickCreate({ permanentLive: true, saveReplay: true })
|
|
|
|
|
|
|
|
const ffmpegCommand = await servers[0].live.sendRTMPStreamInVideo({ videoId: video.uuid })
|
|
|
|
await waitUntilLivePublishedOnAllServers(servers, video.uuid)
|
|
|
|
|
|
|
|
await testLiveVideoResolutions({
|
|
|
|
originServer: servers[0],
|
|
|
|
sqlCommand: sqlCommandServer1,
|
|
|
|
servers,
|
|
|
|
liveVideoId: video.uuid,
|
|
|
|
resolutions: [ 720, 480, 360, 240, 144 ],
|
|
|
|
objectStorage,
|
|
|
|
transcoded: true
|
|
|
|
})
|
|
|
|
|
|
|
|
await stopFfmpeg(ffmpegCommand)
|
|
|
|
|
|
|
|
await waitUntilLiveWaitingOnAllServers(servers, video.uuid)
|
|
|
|
await waitJobs(servers)
|
|
|
|
|
|
|
|
const session = await servers[0].live.findLatestSession({ videoId: video.uuid })
|
|
|
|
expect(session.endingProcessed).to.be.true
|
|
|
|
expect(session.endDate).to.exist
|
|
|
|
expect(session.saveReplay).to.be.true
|
|
|
|
|
|
|
|
const videoLiveDetails = await servers[0].videos.get({ id: video.uuid })
|
|
|
|
const replay = await findExternalSavedVideo(servers[0], videoLiveDetails)
|
|
|
|
|
|
|
|
for (const server of servers) {
|
|
|
|
const video = await server.videos.get({ id: replay.uuid })
|
|
|
|
|
|
|
|
expect(video.files).to.have.lengthOf(0)
|
|
|
|
expect(video.streamingPlaylists).to.have.lengthOf(1)
|
|
|
|
|
|
|
|
const files = video.streamingPlaylists[0].files
|
|
|
|
expect(files).to.have.lengthOf(5)
|
|
|
|
|
|
|
|
for (const file of files) {
|
|
|
|
if (objectStorage) {
|
2023-05-23 04:49:45 -04:00
|
|
|
expectStartWith(file.fileUrl, objectStorage.getMockPlaylistBaseUrl())
|
2023-04-21 09:00:01 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
await makeRawRequest({ url: file.fileUrl, expectedStatus: HttpStatusCode.OK_200 })
|
|
|
|
}
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
before(async function () {
|
|
|
|
this.timeout(120_000)
|
|
|
|
|
|
|
|
servers = await createMultipleServers(2)
|
|
|
|
|
|
|
|
await setAccessTokensToServers(servers)
|
|
|
|
await setDefaultVideoChannel(servers)
|
|
|
|
|
|
|
|
await doubleFollow(servers[0], servers[1])
|
|
|
|
|
|
|
|
sqlCommandServer1 = new SQLCommand(servers[0])
|
|
|
|
|
|
|
|
await servers[0].config.enableRemoteTranscoding()
|
|
|
|
await servers[0].config.enableTranscoding(true, true, true)
|
|
|
|
await servers[0].config.enableLive({ allowReplay: true, resolutions: 'max', transcoding: true })
|
|
|
|
|
|
|
|
const registrationToken = await servers[0].runnerRegistrationTokens.getFirstRegistrationToken()
|
|
|
|
|
2023-05-19 07:33:27 -04:00
|
|
|
peertubeRunner = new PeerTubeRunnerProcess(servers[0])
|
2023-05-05 04:53:04 -04:00
|
|
|
await peertubeRunner.runServer()
|
2023-05-19 07:33:27 -04:00
|
|
|
await peertubeRunner.registerPeerTubeInstance({ registrationToken, runnerName: 'runner' })
|
2023-04-21 09:00:01 -04:00
|
|
|
})
|
|
|
|
|
|
|
|
describe('With lives on local filesystem storage', function () {
|
|
|
|
|
|
|
|
before(async function () {
|
|
|
|
await servers[0].config.enableTranscoding(true, false, true)
|
|
|
|
})
|
|
|
|
|
2023-05-23 04:49:45 -04:00
|
|
|
runSuite()
|
2023-04-21 09:00:01 -04:00
|
|
|
})
|
|
|
|
|
|
|
|
describe('With lives on object storage', function () {
|
|
|
|
if (areMockObjectStorageTestsDisabled()) return
|
|
|
|
|
2023-05-23 04:49:45 -04:00
|
|
|
const objectStorage = new ObjectStorageCommand()
|
|
|
|
|
2023-04-21 09:00:01 -04:00
|
|
|
before(async function () {
|
2023-05-23 04:49:45 -04:00
|
|
|
await objectStorage.prepareDefaultMockBuckets()
|
2023-04-21 09:00:01 -04:00
|
|
|
|
|
|
|
await servers[0].kill()
|
|
|
|
|
2023-05-23 04:49:45 -04:00
|
|
|
await servers[0].run(objectStorage.getDefaultMockConfig())
|
2023-04-21 09:00:01 -04:00
|
|
|
|
|
|
|
// Wait for peertube runner socket reconnection
|
|
|
|
await wait(1500)
|
|
|
|
})
|
|
|
|
|
2023-05-23 04:49:45 -04:00
|
|
|
runSuite({ objectStorage })
|
|
|
|
|
|
|
|
after(async function () {
|
|
|
|
await objectStorage.cleanupMock()
|
|
|
|
})
|
2023-04-21 09:00:01 -04:00
|
|
|
})
|
|
|
|
|
2023-05-04 09:29:34 -04:00
|
|
|
describe('Check cleanup', function () {
|
|
|
|
|
|
|
|
it('Should have an empty cache directory', async function () {
|
2023-05-19 08:35:03 -04:00
|
|
|
await checkPeerTubeRunnerCacheIsEmpty(peertubeRunner)
|
2023-05-04 09:29:34 -04:00
|
|
|
})
|
|
|
|
})
|
|
|
|
|
2023-04-21 09:00:01 -04:00
|
|
|
after(async function () {
|
2023-05-05 07:41:48 -04:00
|
|
|
if (peertubeRunner) {
|
2023-05-19 07:33:27 -04:00
|
|
|
await peertubeRunner.unregisterPeerTubeInstance()
|
2023-05-05 07:41:48 -04:00
|
|
|
peertubeRunner.kill()
|
|
|
|
}
|
2023-04-21 09:00:01 -04:00
|
|
|
|
2023-05-10 02:43:16 -04:00
|
|
|
if (sqlCommandServer1) await sqlCommandServer1.cleanup()
|
|
|
|
|
2023-04-21 09:00:01 -04:00
|
|
|
await cleanupTests(servers)
|
|
|
|
})
|
|
|
|
})
|