2020-11-04 14:16:57 +01:00
|
|
|
/* eslint-disable @typescript-eslint/no-unused-expressions,@typescript-eslint/require-await */
|
|
|
|
|
2022-08-17 15:44:32 +02:00
|
|
|
import { expect } from 'chai'
|
2020-11-04 14:16:57 +01:00
|
|
|
import { FfmpegCommand } from 'fluent-ffmpeg'
|
2022-04-21 09:06:52 +02:00
|
|
|
import { checkLiveCleanup } from '@server/tests/shared'
|
2021-12-17 11:58:15 +01:00
|
|
|
import { wait } from '@shared/core-utils'
|
2022-05-03 11:38:07 +02:00
|
|
|
import { HttpStatusCode, LiveVideoCreate, LiveVideoError, VideoPrivacy, VideoState } from '@shared/models'
|
2020-11-04 14:16:57 +01:00
|
|
|
import {
|
|
|
|
cleanupTests,
|
2021-07-07 11:51:09 +02:00
|
|
|
ConfigCommand,
|
2021-07-16 09:47:51 +02:00
|
|
|
createMultipleServers,
|
2021-07-16 14:27:30 +02:00
|
|
|
doubleFollow,
|
2022-04-21 09:06:52 +02:00
|
|
|
findExternalSavedVideo,
|
2021-07-16 09:47:51 +02:00
|
|
|
PeerTubeServer,
|
2020-11-04 14:16:57 +01:00
|
|
|
setAccessTokensToServers,
|
|
|
|
setDefaultVideoChannel,
|
|
|
|
stopFfmpeg,
|
|
|
|
testFfmpegStreamError,
|
2021-08-17 08:26:20 +02:00
|
|
|
waitJobs,
|
|
|
|
waitUntilLivePublishedOnAllServers,
|
2022-04-21 09:06:52 +02:00
|
|
|
waitUntilLiveReplacedByReplayOnAllServers,
|
|
|
|
waitUntilLiveWaitingOnAllServers
|
2021-12-17 09:29:23 +01:00
|
|
|
} from '@shared/server-commands'
|
2020-11-04 14:16:57 +01:00
|
|
|
|
|
|
|
describe('Save replay setting', function () {
|
2021-07-16 09:47:51 +02:00
|
|
|
let servers: PeerTubeServer[] = []
|
2020-11-04 14:16:57 +01:00
|
|
|
let liveVideoUUID: string
|
|
|
|
let ffmpegCommand: FfmpegCommand
|
|
|
|
|
2023-03-31 09:12:21 +02:00
|
|
|
async function createLiveWrapper (options: { permanent: boolean, replay: boolean, replaySettings?: { privacy: VideoPrivacy } }) {
|
2020-11-04 14:16:57 +01:00
|
|
|
if (liveVideoUUID) {
|
|
|
|
try {
|
2021-07-16 09:04:35 +02:00
|
|
|
await servers[0].videos.remove({ id: liveVideoUUID })
|
2020-11-04 14:16:57 +01:00
|
|
|
await waitJobs(servers)
|
|
|
|
} catch {}
|
|
|
|
}
|
|
|
|
|
|
|
|
const attributes: LiveVideoCreate = {
|
2021-07-16 09:04:35 +02:00
|
|
|
channelId: servers[0].store.channel.id,
|
2020-11-04 14:16:57 +01:00
|
|
|
privacy: VideoPrivacy.PUBLIC,
|
2023-06-29 08:57:19 +02:00
|
|
|
name: 'live'.repeat(30),
|
2022-04-21 09:06:52 +02:00
|
|
|
saveReplay: options.replay,
|
2023-03-31 09:12:21 +02:00
|
|
|
replaySettings: options.replaySettings,
|
2022-04-21 09:06:52 +02:00
|
|
|
permanentLive: options.permanent
|
2020-11-04 14:16:57 +01:00
|
|
|
}
|
|
|
|
|
2021-07-16 09:04:35 +02:00
|
|
|
const { uuid } = await servers[0].live.create({ fields: attributes })
|
2021-07-08 10:18:40 +02:00
|
|
|
return uuid
|
2020-11-04 14:16:57 +01:00
|
|
|
}
|
|
|
|
|
2023-03-31 09:12:21 +02:00
|
|
|
async function publishLive (options: { permanent: boolean, replay: boolean, replaySettings?: { privacy: VideoPrivacy } }) {
|
2022-04-22 09:50:20 +02:00
|
|
|
liveVideoUUID = await createLiveWrapper(options)
|
|
|
|
|
|
|
|
const ffmpegCommand = await servers[0].live.sendRTMPStreamInVideo({ videoId: liveVideoUUID })
|
|
|
|
await waitUntilLivePublishedOnAllServers(servers, liveVideoUUID)
|
|
|
|
|
|
|
|
const liveDetails = await servers[0].videos.get({ id: liveVideoUUID })
|
|
|
|
|
|
|
|
await waitJobs(servers)
|
|
|
|
await checkVideosExist(liveVideoUUID, true, HttpStatusCode.OK_200)
|
|
|
|
|
|
|
|
return { ffmpegCommand, liveDetails }
|
|
|
|
}
|
|
|
|
|
2023-03-31 09:12:21 +02:00
|
|
|
async function publishLiveAndDelete (options: { permanent: boolean, replay: boolean, replaySettings?: { privacy: VideoPrivacy } }) {
|
2022-04-22 09:50:20 +02:00
|
|
|
const { ffmpegCommand, liveDetails } = await publishLive(options)
|
|
|
|
|
|
|
|
await Promise.all([
|
|
|
|
servers[0].videos.remove({ id: liveVideoUUID }),
|
|
|
|
testFfmpegStreamError(ffmpegCommand, true)
|
|
|
|
])
|
|
|
|
|
|
|
|
await waitJobs(servers)
|
|
|
|
await wait(5000)
|
|
|
|
await waitJobs(servers)
|
|
|
|
|
|
|
|
return { liveDetails }
|
|
|
|
}
|
|
|
|
|
2023-03-31 09:12:21 +02:00
|
|
|
async function publishLiveAndBlacklist (options: { permanent: boolean, replay: boolean, replaySettings?: { privacy: VideoPrivacy } }) {
|
2022-04-22 09:50:20 +02:00
|
|
|
const { ffmpegCommand, liveDetails } = await publishLive(options)
|
|
|
|
|
|
|
|
await Promise.all([
|
|
|
|
servers[0].blacklist.add({ videoId: liveVideoUUID, reason: 'bad live', unfederate: true }),
|
|
|
|
testFfmpegStreamError(ffmpegCommand, true)
|
|
|
|
])
|
|
|
|
|
|
|
|
await waitJobs(servers)
|
|
|
|
await wait(5000)
|
|
|
|
await waitJobs(servers)
|
|
|
|
|
|
|
|
return { liveDetails }
|
|
|
|
}
|
|
|
|
|
2021-07-15 10:02:54 +02:00
|
|
|
async function checkVideosExist (videoId: string, existsInList: boolean, expectedStatus?: number) {
|
2020-11-04 14:16:57 +01:00
|
|
|
for (const server of servers) {
|
|
|
|
const length = existsInList ? 1 : 0
|
|
|
|
|
2021-07-16 09:04:35 +02:00
|
|
|
const { data, total } = await server.videos.list()
|
2021-07-15 10:02:54 +02:00
|
|
|
expect(data).to.have.lengthOf(length)
|
|
|
|
expect(total).to.equal(length)
|
2020-11-04 14:16:57 +01:00
|
|
|
|
2021-07-15 10:02:54 +02:00
|
|
|
if (expectedStatus) {
|
2021-07-16 09:04:35 +02:00
|
|
|
await server.videos.get({ id: videoId, expectedStatus })
|
2020-11-04 14:16:57 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
async function checkVideoState (videoId: string, state: VideoState) {
|
|
|
|
for (const server of servers) {
|
2021-07-16 09:04:35 +02:00
|
|
|
const video = await server.videos.get({ id: videoId })
|
2021-07-15 10:02:54 +02:00
|
|
|
expect(video.state.id).to.equal(state)
|
2020-11-04 14:16:57 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-31 09:12:21 +02:00
|
|
|
async function checkVideoPrivacy (videoId: string, privacy: VideoPrivacy) {
|
|
|
|
for (const server of servers) {
|
|
|
|
const video = await server.videos.get({ id: videoId })
|
|
|
|
expect(video.privacy.id).to.equal(privacy)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-04 14:16:57 +01:00
|
|
|
before(async function () {
|
|
|
|
this.timeout(120000)
|
|
|
|
|
2021-07-16 09:47:51 +02:00
|
|
|
servers = await createMultipleServers(2)
|
2020-11-04 14:16:57 +01:00
|
|
|
|
|
|
|
// Get the access tokens
|
|
|
|
await setAccessTokensToServers(servers)
|
|
|
|
await setDefaultVideoChannel(servers)
|
|
|
|
|
|
|
|
// Server 1 and server 2 follow each other
|
|
|
|
await doubleFollow(servers[0], servers[1])
|
|
|
|
|
2021-07-16 09:04:35 +02:00
|
|
|
await servers[0].config.updateCustomSubConfig({
|
2021-07-07 11:51:09 +02:00
|
|
|
newConfig: {
|
|
|
|
live: {
|
|
|
|
enabled: true,
|
|
|
|
allowReplay: true,
|
|
|
|
maxDuration: -1,
|
|
|
|
transcoding: {
|
|
|
|
enabled: false,
|
|
|
|
resolutions: ConfigCommand.getCustomConfigResolutions(true)
|
|
|
|
}
|
2020-11-04 14:16:57 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
describe('With save replay disabled', function () {
|
2022-05-03 11:38:07 +02:00
|
|
|
let sessionStartDateMin: Date
|
|
|
|
let sessionStartDateMax: Date
|
|
|
|
let sessionEndDateMin: Date
|
2020-11-04 14:16:57 +01:00
|
|
|
|
|
|
|
it('Should correctly create and federate the "waiting for stream" live', async function () {
|
2023-05-11 11:25:33 +02:00
|
|
|
this.timeout(40000)
|
2020-11-04 14:16:57 +01:00
|
|
|
|
2022-04-21 09:06:52 +02:00
|
|
|
liveVideoUUID = await createLiveWrapper({ permanent: false, replay: false })
|
2020-11-04 14:16:57 +01:00
|
|
|
|
|
|
|
await waitJobs(servers)
|
|
|
|
|
2020-12-08 21:16:10 +01:00
|
|
|
await checkVideosExist(liveVideoUUID, false, HttpStatusCode.OK_200)
|
2020-11-04 14:16:57 +01:00
|
|
|
await checkVideoState(liveVideoUUID, VideoState.WAITING_FOR_LIVE)
|
|
|
|
})
|
|
|
|
|
|
|
|
it('Should correctly have updated the live and federated it when streaming in the live', async function () {
|
2023-05-24 15:27:15 +02:00
|
|
|
this.timeout(120000)
|
2020-11-04 14:16:57 +01:00
|
|
|
|
2021-07-16 09:04:35 +02:00
|
|
|
ffmpegCommand = await servers[0].live.sendRTMPStreamInVideo({ videoId: liveVideoUUID })
|
2021-02-18 14:44:12 +01:00
|
|
|
|
2022-05-03 11:38:07 +02:00
|
|
|
sessionStartDateMin = new Date()
|
2021-08-17 08:26:20 +02:00
|
|
|
await waitUntilLivePublishedOnAllServers(servers, liveVideoUUID)
|
2022-05-03 11:38:07 +02:00
|
|
|
sessionStartDateMax = new Date()
|
2020-11-04 14:16:57 +01:00
|
|
|
|
|
|
|
await waitJobs(servers)
|
|
|
|
|
2020-12-08 21:16:10 +01:00
|
|
|
await checkVideosExist(liveVideoUUID, true, HttpStatusCode.OK_200)
|
2020-11-04 14:16:57 +01:00
|
|
|
await checkVideoState(liveVideoUUID, VideoState.PUBLISHED)
|
|
|
|
})
|
|
|
|
|
|
|
|
it('Should correctly delete the video files after the stream ended', async function () {
|
2023-05-24 15:27:15 +02:00
|
|
|
this.timeout(120000)
|
2020-11-04 14:16:57 +01:00
|
|
|
|
2022-05-03 11:38:07 +02:00
|
|
|
sessionEndDateMin = new Date()
|
2020-11-04 14:16:57 +01:00
|
|
|
await stopFfmpeg(ffmpegCommand)
|
|
|
|
|
2021-02-18 14:44:12 +01:00
|
|
|
for (const server of servers) {
|
2021-07-16 09:04:35 +02:00
|
|
|
await server.live.waitUntilEnded({ videoId: liveVideoUUID })
|
2021-02-18 14:44:12 +01:00
|
|
|
}
|
2020-11-04 14:16:57 +01:00
|
|
|
await waitJobs(servers)
|
|
|
|
|
|
|
|
// Live still exist, but cannot be played anymore
|
2020-12-08 21:16:10 +01:00
|
|
|
await checkVideosExist(liveVideoUUID, false, HttpStatusCode.OK_200)
|
2020-11-04 14:16:57 +01:00
|
|
|
await checkVideoState(liveVideoUUID, VideoState.LIVE_ENDED)
|
|
|
|
|
|
|
|
// No resolutions saved since we did not save replay
|
2022-10-25 14:18:59 +02:00
|
|
|
await checkLiveCleanup({ server: servers[0], videoUUID: liveVideoUUID, permanent: false })
|
2020-11-04 14:16:57 +01:00
|
|
|
})
|
|
|
|
|
2022-05-03 11:38:07 +02:00
|
|
|
it('Should have appropriate ended session', async function () {
|
|
|
|
const { data, total } = await servers[0].live.listSessions({ videoId: liveVideoUUID })
|
|
|
|
expect(total).to.equal(1)
|
|
|
|
expect(data).to.have.lengthOf(1)
|
|
|
|
|
|
|
|
const session = data[0]
|
|
|
|
|
|
|
|
const startDate = new Date(session.startDate)
|
|
|
|
expect(startDate).to.be.above(sessionStartDateMin)
|
|
|
|
expect(startDate).to.be.below(sessionStartDateMax)
|
|
|
|
|
|
|
|
expect(session.endDate).to.exist
|
|
|
|
expect(new Date(session.endDate)).to.be.above(sessionEndDateMin)
|
|
|
|
|
2022-07-22 15:22:21 +02:00
|
|
|
expect(session.saveReplay).to.be.false
|
2022-05-03 11:38:07 +02:00
|
|
|
expect(session.error).to.not.exist
|
|
|
|
expect(session.replayVideo).to.not.exist
|
|
|
|
})
|
|
|
|
|
2020-11-04 14:16:57 +01:00
|
|
|
it('Should correctly terminate the stream on blacklist and delete the live', async function () {
|
2023-05-24 15:27:15 +02:00
|
|
|
this.timeout(120000)
|
2020-11-04 14:16:57 +01:00
|
|
|
|
2022-04-22 09:50:20 +02:00
|
|
|
await publishLiveAndBlacklist({ permanent: false, replay: false })
|
2020-11-04 14:16:57 +01:00
|
|
|
|
|
|
|
await checkVideosExist(liveVideoUUID, false)
|
|
|
|
|
2021-07-16 09:04:35 +02:00
|
|
|
await servers[0].videos.get({ id: liveVideoUUID, expectedStatus: HttpStatusCode.UNAUTHORIZED_401 })
|
|
|
|
await servers[1].videos.get({ id: liveVideoUUID, expectedStatus: HttpStatusCode.NOT_FOUND_404 })
|
2020-11-04 14:16:57 +01:00
|
|
|
|
2021-02-19 14:30:00 +01:00
|
|
|
await wait(5000)
|
|
|
|
await waitJobs(servers)
|
2022-10-25 14:18:59 +02:00
|
|
|
await checkLiveCleanup({ server: servers[0], videoUUID: liveVideoUUID, permanent: false })
|
2020-11-04 14:16:57 +01:00
|
|
|
})
|
|
|
|
|
2022-05-03 11:38:07 +02:00
|
|
|
it('Should have blacklisted session error', async function () {
|
|
|
|
const session = await servers[0].live.findLatestSession({ videoId: liveVideoUUID })
|
|
|
|
expect(session.startDate).to.exist
|
|
|
|
expect(session.endDate).to.exist
|
|
|
|
|
|
|
|
expect(session.error).to.equal(LiveVideoError.BLACKLISTED)
|
|
|
|
expect(session.replayVideo).to.not.exist
|
|
|
|
})
|
|
|
|
|
2020-11-04 14:16:57 +01:00
|
|
|
it('Should correctly terminate the stream on delete and delete the video', async function () {
|
2023-05-24 15:27:15 +02:00
|
|
|
this.timeout(120000)
|
2020-11-04 14:16:57 +01:00
|
|
|
|
2022-04-22 09:50:20 +02:00
|
|
|
await publishLiveAndDelete({ permanent: false, replay: false })
|
2020-11-04 14:16:57 +01:00
|
|
|
|
2020-12-08 21:16:10 +01:00
|
|
|
await checkVideosExist(liveVideoUUID, false, HttpStatusCode.NOT_FOUND_404)
|
2022-10-25 14:18:59 +02:00
|
|
|
await checkLiveCleanup({ server: servers[0], videoUUID: liveVideoUUID, permanent: false })
|
2020-11-04 14:16:57 +01:00
|
|
|
})
|
|
|
|
})
|
|
|
|
|
2022-04-21 09:06:52 +02:00
|
|
|
describe('With save replay enabled on non permanent live', function () {
|
2020-11-04 14:16:57 +01:00
|
|
|
|
|
|
|
it('Should correctly create and federate the "waiting for stream" live', async function () {
|
2023-05-24 15:27:15 +02:00
|
|
|
this.timeout(120000)
|
2020-11-04 14:16:57 +01:00
|
|
|
|
2023-03-31 09:12:21 +02:00
|
|
|
liveVideoUUID = await createLiveWrapper({ permanent: false, replay: true, replaySettings: { privacy: VideoPrivacy.UNLISTED } })
|
2020-11-04 14:16:57 +01:00
|
|
|
|
|
|
|
await waitJobs(servers)
|
|
|
|
|
2020-12-08 21:16:10 +01:00
|
|
|
await checkVideosExist(liveVideoUUID, false, HttpStatusCode.OK_200)
|
2020-11-04 14:16:57 +01:00
|
|
|
await checkVideoState(liveVideoUUID, VideoState.WAITING_FOR_LIVE)
|
2023-03-31 09:12:21 +02:00
|
|
|
await checkVideoPrivacy(liveVideoUUID, VideoPrivacy.PUBLIC)
|
2020-11-04 14:16:57 +01:00
|
|
|
})
|
|
|
|
|
|
|
|
it('Should correctly have updated the live and federated it when streaming in the live', async function () {
|
2023-05-24 15:27:15 +02:00
|
|
|
this.timeout(120000)
|
2020-11-04 14:16:57 +01:00
|
|
|
|
2021-07-16 09:04:35 +02:00
|
|
|
ffmpegCommand = await servers[0].live.sendRTMPStreamInVideo({ videoId: liveVideoUUID })
|
2021-08-17 08:26:20 +02:00
|
|
|
await waitUntilLivePublishedOnAllServers(servers, liveVideoUUID)
|
2020-11-04 14:16:57 +01:00
|
|
|
|
|
|
|
await waitJobs(servers)
|
|
|
|
|
2020-12-08 21:16:10 +01:00
|
|
|
await checkVideosExist(liveVideoUUID, true, HttpStatusCode.OK_200)
|
2020-11-04 14:16:57 +01:00
|
|
|
await checkVideoState(liveVideoUUID, VideoState.PUBLISHED)
|
2023-03-31 09:12:21 +02:00
|
|
|
await checkVideoPrivacy(liveVideoUUID, VideoPrivacy.PUBLIC)
|
2020-11-04 14:16:57 +01:00
|
|
|
})
|
|
|
|
|
|
|
|
it('Should correctly have saved the live and federated it after the streaming', async function () {
|
2023-05-24 15:27:15 +02:00
|
|
|
this.timeout(120000)
|
2020-11-04 14:16:57 +01:00
|
|
|
|
2022-07-22 15:22:21 +02:00
|
|
|
const session = await servers[0].live.findLatestSession({ videoId: liveVideoUUID })
|
|
|
|
expect(session.endDate).to.not.exist
|
|
|
|
expect(session.endingProcessed).to.be.false
|
|
|
|
expect(session.saveReplay).to.be.true
|
2023-03-31 09:12:21 +02:00
|
|
|
expect(session.replaySettings).to.exist
|
|
|
|
expect(session.replaySettings.privacy).to.equal(VideoPrivacy.UNLISTED)
|
2022-07-22 15:22:21 +02:00
|
|
|
|
2020-11-04 14:16:57 +01:00
|
|
|
await stopFfmpeg(ffmpegCommand)
|
|
|
|
|
2022-04-21 09:06:52 +02:00
|
|
|
await waitUntilLiveReplacedByReplayOnAllServers(servers, liveVideoUUID)
|
2020-11-04 14:16:57 +01:00
|
|
|
await waitJobs(servers)
|
|
|
|
|
|
|
|
// Live has been transcoded
|
2023-03-31 09:12:21 +02:00
|
|
|
await checkVideosExist(liveVideoUUID, false, HttpStatusCode.OK_200)
|
2020-11-04 14:16:57 +01:00
|
|
|
await checkVideoState(liveVideoUUID, VideoState.PUBLISHED)
|
2023-03-31 09:12:21 +02:00
|
|
|
await checkVideoPrivacy(liveVideoUUID, VideoPrivacy.UNLISTED)
|
2020-11-04 14:16:57 +01:00
|
|
|
})
|
|
|
|
|
2022-05-03 11:38:07 +02:00
|
|
|
it('Should find the replay live session', async function () {
|
|
|
|
const session = await servers[0].live.getReplaySession({ videoId: liveVideoUUID })
|
|
|
|
|
|
|
|
expect(session).to.exist
|
|
|
|
|
|
|
|
expect(session.startDate).to.exist
|
|
|
|
expect(session.endDate).to.exist
|
|
|
|
|
|
|
|
expect(session.error).to.not.exist
|
2022-07-22 15:22:21 +02:00
|
|
|
expect(session.saveReplay).to.be.true
|
|
|
|
expect(session.endingProcessed).to.be.true
|
2023-03-31 09:12:21 +02:00
|
|
|
expect(session.replaySettings).to.exist
|
|
|
|
expect(session.replaySettings.privacy).to.equal(VideoPrivacy.UNLISTED)
|
2022-05-03 11:38:07 +02:00
|
|
|
|
|
|
|
expect(session.replayVideo).to.exist
|
|
|
|
expect(session.replayVideo.id).to.exist
|
|
|
|
expect(session.replayVideo.shortUUID).to.exist
|
|
|
|
expect(session.replayVideo.uuid).to.equal(liveVideoUUID)
|
|
|
|
})
|
|
|
|
|
2020-11-04 14:16:57 +01:00
|
|
|
it('Should update the saved live and correctly federate the updated attributes', async function () {
|
2023-05-24 15:27:15 +02:00
|
|
|
this.timeout(120000)
|
2020-11-04 14:16:57 +01:00
|
|
|
|
2023-03-31 09:12:21 +02:00
|
|
|
await servers[0].videos.update({ id: liveVideoUUID, attributes: { name: 'video updated', privacy: VideoPrivacy.PUBLIC } })
|
2020-11-04 14:16:57 +01:00
|
|
|
await waitJobs(servers)
|
|
|
|
|
|
|
|
for (const server of servers) {
|
2021-07-16 09:04:35 +02:00
|
|
|
const video = await server.videos.get({ id: liveVideoUUID })
|
2021-07-15 10:02:54 +02:00
|
|
|
expect(video.name).to.equal('video updated')
|
|
|
|
expect(video.isLive).to.be.false
|
2023-03-31 09:12:21 +02:00
|
|
|
expect(video.privacy.id).to.equal(VideoPrivacy.PUBLIC)
|
2020-11-04 14:16:57 +01:00
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
it('Should have cleaned up the live files', async function () {
|
2022-10-25 14:18:59 +02:00
|
|
|
await checkLiveCleanup({ server: servers[0], videoUUID: liveVideoUUID, permanent: false, savedResolutions: [ 720 ] })
|
2020-11-04 14:16:57 +01:00
|
|
|
})
|
|
|
|
|
|
|
|
it('Should correctly terminate the stream on blacklist and blacklist the saved replay video', async function () {
|
2022-07-08 11:51:41 +02:00
|
|
|
this.timeout(120000)
|
2020-11-04 14:16:57 +01:00
|
|
|
|
2023-03-31 09:12:21 +02:00
|
|
|
await publishLiveAndBlacklist({ permanent: false, replay: true, replaySettings: { privacy: VideoPrivacy.PUBLIC } })
|
2020-11-04 14:16:57 +01:00
|
|
|
|
|
|
|
await checkVideosExist(liveVideoUUID, false)
|
|
|
|
|
2021-07-16 09:04:35 +02:00
|
|
|
await servers[0].videos.get({ id: liveVideoUUID, expectedStatus: HttpStatusCode.UNAUTHORIZED_401 })
|
|
|
|
await servers[1].videos.get({ id: liveVideoUUID, expectedStatus: HttpStatusCode.NOT_FOUND_404 })
|
2020-11-04 14:16:57 +01:00
|
|
|
|
2021-02-19 14:30:00 +01:00
|
|
|
await wait(5000)
|
|
|
|
await waitJobs(servers)
|
2022-10-25 14:18:59 +02:00
|
|
|
await checkLiveCleanup({ server: servers[0], videoUUID: liveVideoUUID, permanent: false, savedResolutions: [ 720 ] })
|
2020-11-04 14:16:57 +01:00
|
|
|
})
|
|
|
|
|
|
|
|
it('Should correctly terminate the stream on delete and delete the video', async function () {
|
2023-05-24 15:27:15 +02:00
|
|
|
this.timeout(120000)
|
2020-11-04 14:16:57 +01:00
|
|
|
|
2023-03-31 09:12:21 +02:00
|
|
|
await publishLiveAndDelete({ permanent: false, replay: true, replaySettings: { privacy: VideoPrivacy.PUBLIC } })
|
2020-11-04 14:16:57 +01:00
|
|
|
|
2020-12-08 21:16:10 +01:00
|
|
|
await checkVideosExist(liveVideoUUID, false, HttpStatusCode.NOT_FOUND_404)
|
2022-10-25 14:18:59 +02:00
|
|
|
await checkLiveCleanup({ server: servers[0], videoUUID: liveVideoUUID, permanent: false })
|
2022-04-21 09:06:52 +02:00
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
describe('With save replay enabled on permanent live', function () {
|
|
|
|
let lastReplayUUID: string
|
|
|
|
|
2023-03-31 09:12:21 +02:00
|
|
|
describe('With a first live and its replay', function () {
|
2022-04-21 09:06:52 +02:00
|
|
|
|
2023-03-31 09:12:21 +02:00
|
|
|
it('Should correctly create and federate the "waiting for stream" live', async function () {
|
2023-05-24 15:27:15 +02:00
|
|
|
this.timeout(120000)
|
2022-04-21 09:06:52 +02:00
|
|
|
|
2023-03-31 09:12:21 +02:00
|
|
|
liveVideoUUID = await createLiveWrapper({ permanent: true, replay: true, replaySettings: { privacy: VideoPrivacy.UNLISTED } })
|
2022-04-21 09:06:52 +02:00
|
|
|
|
2023-03-31 09:12:21 +02:00
|
|
|
await waitJobs(servers)
|
2022-04-21 09:06:52 +02:00
|
|
|
|
2023-03-31 09:12:21 +02:00
|
|
|
await checkVideosExist(liveVideoUUID, false, HttpStatusCode.OK_200)
|
|
|
|
await checkVideoState(liveVideoUUID, VideoState.WAITING_FOR_LIVE)
|
|
|
|
await checkVideoPrivacy(liveVideoUUID, VideoPrivacy.PUBLIC)
|
|
|
|
})
|
2022-04-21 09:06:52 +02:00
|
|
|
|
2023-03-31 09:12:21 +02:00
|
|
|
it('Should correctly have updated the live and federated it when streaming in the live', async function () {
|
2023-05-24 15:27:15 +02:00
|
|
|
this.timeout(120000)
|
2022-04-21 09:06:52 +02:00
|
|
|
|
2023-03-31 09:12:21 +02:00
|
|
|
ffmpegCommand = await servers[0].live.sendRTMPStreamInVideo({ videoId: liveVideoUUID })
|
|
|
|
await waitUntilLivePublishedOnAllServers(servers, liveVideoUUID)
|
2022-04-21 09:06:52 +02:00
|
|
|
|
2023-03-31 09:12:21 +02:00
|
|
|
await waitJobs(servers)
|
2022-04-21 09:06:52 +02:00
|
|
|
|
2023-03-31 09:12:21 +02:00
|
|
|
await checkVideosExist(liveVideoUUID, true, HttpStatusCode.OK_200)
|
|
|
|
await checkVideoState(liveVideoUUID, VideoState.PUBLISHED)
|
|
|
|
await checkVideoPrivacy(liveVideoUUID, VideoPrivacy.PUBLIC)
|
|
|
|
})
|
2022-04-21 09:06:52 +02:00
|
|
|
|
2023-03-31 09:12:21 +02:00
|
|
|
it('Should correctly have saved the live and federated it after the streaming', async function () {
|
2023-05-24 15:27:15 +02:00
|
|
|
this.timeout(120000)
|
2022-04-21 09:06:52 +02:00
|
|
|
|
2023-03-31 09:12:21 +02:00
|
|
|
const liveDetails = await servers[0].videos.get({ id: liveVideoUUID })
|
2022-04-21 09:06:52 +02:00
|
|
|
|
2023-03-31 09:12:21 +02:00
|
|
|
await stopFfmpeg(ffmpegCommand)
|
2022-04-21 09:06:52 +02:00
|
|
|
|
2023-03-31 09:12:21 +02:00
|
|
|
await waitUntilLiveWaitingOnAllServers(servers, liveVideoUUID)
|
|
|
|
await waitJobs(servers)
|
2022-04-21 09:06:52 +02:00
|
|
|
|
2023-03-31 09:12:21 +02:00
|
|
|
const video = await findExternalSavedVideo(servers[0], liveDetails)
|
|
|
|
expect(video).to.exist
|
2022-04-21 09:06:52 +02:00
|
|
|
|
2023-03-31 09:12:21 +02:00
|
|
|
for (const server of servers) {
|
|
|
|
await server.videos.get({ id: video.uuid })
|
|
|
|
}
|
2022-04-21 09:06:52 +02:00
|
|
|
|
2023-03-31 09:12:21 +02:00
|
|
|
lastReplayUUID = video.uuid
|
|
|
|
})
|
2022-05-03 11:38:07 +02:00
|
|
|
|
2023-03-31 09:12:21 +02:00
|
|
|
it('Should have appropriate ended session and replay live session', async function () {
|
|
|
|
const { data, total } = await servers[0].live.listSessions({ videoId: liveVideoUUID })
|
|
|
|
expect(total).to.equal(1)
|
|
|
|
expect(data).to.have.lengthOf(1)
|
2022-05-03 11:38:07 +02:00
|
|
|
|
2023-03-31 09:12:21 +02:00
|
|
|
const sessionFromLive = data[0]
|
|
|
|
const sessionFromReplay = await servers[0].live.getReplaySession({ videoId: lastReplayUUID })
|
2022-05-03 11:38:07 +02:00
|
|
|
|
2023-03-31 09:12:21 +02:00
|
|
|
for (const session of [ sessionFromLive, sessionFromReplay ]) {
|
|
|
|
expect(session.startDate).to.exist
|
|
|
|
expect(session.endDate).to.exist
|
2022-05-03 11:38:07 +02:00
|
|
|
|
2023-03-31 09:12:21 +02:00
|
|
|
expect(session.replaySettings).to.exist
|
|
|
|
expect(session.replaySettings.privacy).to.equal(VideoPrivacy.UNLISTED)
|
2022-05-03 11:38:07 +02:00
|
|
|
|
2023-03-31 09:12:21 +02:00
|
|
|
expect(session.error).to.not.exist
|
|
|
|
|
|
|
|
expect(session.replayVideo).to.exist
|
|
|
|
expect(session.replayVideo.id).to.exist
|
|
|
|
expect(session.replayVideo.shortUUID).to.exist
|
|
|
|
expect(session.replayVideo.uuid).to.equal(lastReplayUUID)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
it('Should have the first live replay with correct settings', async function () {
|
|
|
|
await checkVideosExist(lastReplayUUID, false, HttpStatusCode.OK_200)
|
|
|
|
await checkVideoState(lastReplayUUID, VideoState.PUBLISHED)
|
|
|
|
await checkVideoPrivacy(lastReplayUUID, VideoPrivacy.UNLISTED)
|
|
|
|
})
|
2022-04-21 09:06:52 +02:00
|
|
|
})
|
|
|
|
|
2023-03-31 09:12:21 +02:00
|
|
|
describe('With a second live and its replay', function () {
|
2023-05-11 11:25:33 +02:00
|
|
|
|
2023-03-31 09:12:21 +02:00
|
|
|
it('Should update the replay settings', async function () {
|
2023-05-11 11:25:33 +02:00
|
|
|
await servers[0].live.update({ videoId: liveVideoUUID, fields: { replaySettings: { privacy: VideoPrivacy.PUBLIC } } })
|
2023-03-31 09:12:21 +02:00
|
|
|
await waitJobs(servers)
|
2023-05-11 11:25:33 +02:00
|
|
|
|
2023-03-31 09:12:21 +02:00
|
|
|
const live = await servers[0].live.get({ videoId: liveVideoUUID })
|
2022-04-21 09:06:52 +02:00
|
|
|
|
2023-03-31 09:12:21 +02:00
|
|
|
expect(live.saveReplay).to.be.true
|
|
|
|
expect(live.replaySettings).to.exist
|
|
|
|
expect(live.replaySettings.privacy).to.equal(VideoPrivacy.PUBLIC)
|
2022-04-21 09:06:52 +02:00
|
|
|
|
2023-03-31 09:12:21 +02:00
|
|
|
})
|
2022-04-21 09:06:52 +02:00
|
|
|
|
2023-03-31 09:12:21 +02:00
|
|
|
it('Should correctly have updated the live and federated it when streaming in the live', async function () {
|
2023-05-24 15:27:15 +02:00
|
|
|
this.timeout(120000)
|
2022-04-21 09:06:52 +02:00
|
|
|
|
2023-03-31 09:12:21 +02:00
|
|
|
ffmpegCommand = await servers[0].live.sendRTMPStreamInVideo({ videoId: liveVideoUUID })
|
|
|
|
await waitUntilLivePublishedOnAllServers(servers, liveVideoUUID)
|
2022-04-21 09:06:52 +02:00
|
|
|
|
2023-03-31 09:12:21 +02:00
|
|
|
await waitJobs(servers)
|
2022-04-21 09:06:52 +02:00
|
|
|
|
2023-03-31 09:12:21 +02:00
|
|
|
await checkVideosExist(liveVideoUUID, true, HttpStatusCode.OK_200)
|
|
|
|
await checkVideoState(liveVideoUUID, VideoState.PUBLISHED)
|
|
|
|
await checkVideoPrivacy(liveVideoUUID, VideoPrivacy.PUBLIC)
|
|
|
|
})
|
2022-04-21 09:06:52 +02:00
|
|
|
|
2023-03-31 09:12:21 +02:00
|
|
|
it('Should correctly have saved the live and federated it after the streaming', async function () {
|
2023-05-24 15:27:15 +02:00
|
|
|
this.timeout(120000)
|
2023-05-11 11:25:33 +02:00
|
|
|
|
2023-03-31 09:12:21 +02:00
|
|
|
const liveDetails = await servers[0].videos.get({ id: liveVideoUUID })
|
2022-04-21 09:06:52 +02:00
|
|
|
|
2023-03-31 09:12:21 +02:00
|
|
|
await stopFfmpeg(ffmpegCommand)
|
2022-04-21 09:06:52 +02:00
|
|
|
|
2023-03-31 09:12:21 +02:00
|
|
|
await waitUntilLiveWaitingOnAllServers(servers, liveVideoUUID)
|
|
|
|
await waitJobs(servers)
|
|
|
|
|
|
|
|
const video = await findExternalSavedVideo(servers[0], liveDetails)
|
|
|
|
expect(video).to.exist
|
|
|
|
|
|
|
|
for (const server of servers) {
|
|
|
|
await server.videos.get({ id: video.uuid })
|
|
|
|
}
|
|
|
|
|
|
|
|
lastReplayUUID = video.uuid
|
|
|
|
})
|
|
|
|
|
|
|
|
it('Should have appropriate ended session and replay live session', async function () {
|
|
|
|
const { data, total } = await servers[0].live.listSessions({ videoId: liveVideoUUID })
|
|
|
|
expect(total).to.equal(2)
|
|
|
|
expect(data).to.have.lengthOf(2)
|
|
|
|
|
|
|
|
const sessionFromLive = data[1]
|
|
|
|
const sessionFromReplay = await servers[0].live.getReplaySession({ videoId: lastReplayUUID })
|
|
|
|
|
|
|
|
for (const session of [ sessionFromLive, sessionFromReplay ]) {
|
|
|
|
expect(session.startDate).to.exist
|
|
|
|
expect(session.endDate).to.exist
|
|
|
|
|
|
|
|
expect(session.replaySettings).to.exist
|
|
|
|
expect(session.replaySettings.privacy).to.equal(VideoPrivacy.PUBLIC)
|
|
|
|
|
|
|
|
expect(session.error).to.not.exist
|
|
|
|
|
|
|
|
expect(session.replayVideo).to.exist
|
|
|
|
expect(session.replayVideo.id).to.exist
|
|
|
|
expect(session.replayVideo.shortUUID).to.exist
|
|
|
|
expect(session.replayVideo.uuid).to.equal(lastReplayUUID)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
it('Should have the first live replay with correct settings', async function () {
|
|
|
|
await checkVideosExist(lastReplayUUID, true, HttpStatusCode.OK_200)
|
|
|
|
await checkVideoState(lastReplayUUID, VideoState.PUBLISHED)
|
|
|
|
await checkVideoPrivacy(lastReplayUUID, VideoPrivacy.PUBLIC)
|
|
|
|
})
|
|
|
|
|
|
|
|
it('Should have cleaned up the live files', async function () {
|
|
|
|
await checkLiveCleanup({ server: servers[0], videoUUID: liveVideoUUID, permanent: false })
|
|
|
|
})
|
|
|
|
|
|
|
|
it('Should correctly terminate the stream on blacklist and blacklist the saved replay video', async function () {
|
|
|
|
this.timeout(120000)
|
|
|
|
|
|
|
|
await servers[0].videos.remove({ id: lastReplayUUID })
|
|
|
|
const { liveDetails } = await publishLiveAndBlacklist({
|
|
|
|
permanent: true,
|
|
|
|
replay: true,
|
|
|
|
replaySettings: { privacy: VideoPrivacy.PUBLIC }
|
|
|
|
})
|
|
|
|
|
|
|
|
const replay = await findExternalSavedVideo(servers[0], liveDetails)
|
|
|
|
expect(replay).to.exist
|
|
|
|
|
|
|
|
for (const videoId of [ liveVideoUUID, replay.uuid ]) {
|
|
|
|
await checkVideosExist(videoId, false)
|
|
|
|
|
|
|
|
await servers[0].videos.get({ id: videoId, expectedStatus: HttpStatusCode.UNAUTHORIZED_401 })
|
|
|
|
await servers[1].videos.get({ id: videoId, expectedStatus: HttpStatusCode.NOT_FOUND_404 })
|
|
|
|
}
|
|
|
|
|
|
|
|
await checkLiveCleanup({ server: servers[0], videoUUID: liveVideoUUID, permanent: false })
|
|
|
|
})
|
|
|
|
|
|
|
|
it('Should correctly terminate the stream on delete and not save the video', async function () {
|
2023-05-24 15:27:15 +02:00
|
|
|
this.timeout(120000)
|
2023-03-31 09:12:21 +02:00
|
|
|
|
|
|
|
const { liveDetails } = await publishLiveAndDelete({
|
|
|
|
permanent: true,
|
|
|
|
replay: true,
|
|
|
|
replaySettings: { privacy: VideoPrivacy.PUBLIC }
|
|
|
|
})
|
|
|
|
|
|
|
|
const replay = await findExternalSavedVideo(servers[0], liveDetails)
|
|
|
|
expect(replay).to.not.exist
|
|
|
|
|
|
|
|
await checkVideosExist(liveVideoUUID, false, HttpStatusCode.NOT_FOUND_404)
|
|
|
|
await checkLiveCleanup({ server: servers[0], videoUUID: liveVideoUUID, permanent: false })
|
|
|
|
})
|
2020-11-04 14:16:57 +01:00
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
after(async function () {
|
|
|
|
await cleanupTests(servers)
|
|
|
|
})
|
|
|
|
})
|