/* eslint-disable @typescript-eslint/no-unused-expressions,@typescript-eslint/require-await */
-import 'mocha'
-import * as chai from 'chai'
-import { expectStartWith } from '@server/tests/shared'
-import { areObjectStorageTestsDisabled } from '@shared/core-utils'
-import { HttpStatusCode, LiveVideoCreate, VideoFile, VideoPrivacy } from '@shared/models'
+import { expect } from 'chai'
+import { expectStartWith, MockObjectStorageProxy, testVideoResolutions } from '@server/tests/shared'
+import { areMockObjectStorageTestsDisabled } from '@shared/core-utils'
+import { HttpStatusCode, LiveVideoCreate, VideoPrivacy } from '@shared/models'
import {
createMultipleServers,
doubleFollow,
waitUntilLiveWaitingOnAllServers
} from '@shared/server-commands'
-const expect = chai.expect
-
async function createLive (server: PeerTubeServer, permanent: boolean) {
const attributes: LiveVideoCreate = {
channelId: server.store.channel.id,
privacy: VideoPrivacy.PUBLIC,
name: 'my super live',
saveReplay: true,
+ replaySettings: { privacy: VideoPrivacy.PUBLIC },
permanentLive: permanent
}
return uuid
}
-async function checkFiles (files: VideoFile[]) {
- for (const file of files) {
- expectStartWith(file.fileUrl, ObjectStorageCommand.getPlaylistBaseUrl())
+async function checkFilesExist (servers: PeerTubeServer[], videoUUID: string, numberOfFiles: number) {
+ for (const server of servers) {
+ const video = await server.videos.get({ id: videoUUID })
- await makeRawRequest(file.fileUrl, HttpStatusCode.OK_200)
- }
-}
+ expect(video.files).to.have.lengthOf(0)
+ expect(video.streamingPlaylists).to.have.lengthOf(1)
-async function getFiles (server: PeerTubeServer, videoUUID: string) {
- const video = await server.videos.get({ id: videoUUID })
+ const files = video.streamingPlaylists[0].files
+ expect(files).to.have.lengthOf(numberOfFiles)
- expect(video.files).to.have.lengthOf(0)
- expect(video.streamingPlaylists).to.have.lengthOf(1)
+ for (const file of files) {
+ expectStartWith(file.fileUrl, ObjectStorageCommand.getMockPlaylistBaseUrl())
- return video.streamingPlaylists[0].files
+ await makeRawRequest({ url: file.fileUrl, expectedStatus: HttpStatusCode.OK_200 })
+ }
+ }
}
-async function streamAndEnd (servers: PeerTubeServer[], liveUUID: string) {
- const ffmpegCommand = await servers[0].live.sendRTMPStreamInVideo({ videoId: liveUUID })
- await waitUntilLivePublishedOnAllServers(servers, liveUUID)
-
- const videoLiveDetails = await servers[0].videos.get({ id: liveUUID })
- const liveDetails = await servers[0].live.get({ videoId: liveUUID })
-
- await stopFfmpeg(ffmpegCommand)
+async function checkFilesCleanup (server: PeerTubeServer, videoUUID: string, resolutions: number[]) {
+ const resolutionFiles = resolutions.map((_value, i) => `${i}.m3u8`)
- if (liveDetails.permanentLive) {
- await waitUntilLiveWaitingOnAllServers(servers, liveUUID)
- } else {
- await waitUntilLiveReplacedByReplayOnAllServers(servers, liveUUID)
+ for (const playlistName of [ 'master.m3u8' ].concat(resolutionFiles)) {
+ await server.live.getPlaylistFile({
+ videoUUID,
+ playlistName,
+ expectedStatus: HttpStatusCode.NOT_FOUND_404,
+ objectStorage: true
+ })
}
- await waitJobs(servers)
-
- return { videoLiveDetails, liveDetails }
+ await server.live.getSegmentFile({
+ videoUUID,
+ playlistNumber: 0,
+ segment: 0,
+ objectStorage: true,
+ expectedStatus: HttpStatusCode.NOT_FOUND_404
+ })
}
describe('Object storage for lives', function () {
- if (areObjectStorageTestsDisabled()) return
+ if (areMockObjectStorageTestsDisabled()) return
let servers: PeerTubeServer[]
before(async function () {
this.timeout(120000)
- await ObjectStorageCommand.prepareDefaultBuckets()
+ await ObjectStorageCommand.prepareDefaultMockBuckets()
- servers = await createMultipleServers(2, ObjectStorageCommand.getDefaultConfig())
+ servers = await createMultipleServers(2, ObjectStorageCommand.getDefaultMockConfig())
await setAccessTokensToServers(servers)
await setDefaultVideoChannel(servers)
await servers[0].config.enableTranscoding()
})
- describe('Without live transcoding', async function () {
+ describe('Without live transcoding', function () {
let videoUUID: string
before(async function () {
videoUUID = await createLive(servers[0], false)
})
- it('Should create a live and save the replay on object storage', async function () {
+ it('Should create a live and publish it on object storage', async function () {
this.timeout(220000)
- await streamAndEnd(servers, videoUUID)
+ const ffmpegCommand = await servers[0].live.sendRTMPStreamInVideo({ videoId: videoUUID })
+ await waitUntilLivePublishedOnAllServers(servers, videoUUID)
- for (const server of servers) {
- const files = await getFiles(server, videoUUID)
- expect(files).to.have.lengthOf(1)
+ await testVideoResolutions({
+ originServer: servers[0],
+ servers,
+ liveVideoId: videoUUID,
+ resolutions: [ 720 ],
+ transcoded: false,
+ objectStorage: true
+ })
- await checkFiles(files)
- }
+ await stopFfmpeg(ffmpegCommand)
+ })
+
+ it('Should have saved the replay on object storage', async function () {
+ this.timeout(220000)
+
+ await waitUntilLiveReplacedByReplayOnAllServers(servers, videoUUID)
+ await waitJobs(servers)
+
+ await checkFilesExist(servers, videoUUID, 1)
+ })
+
+ it('Should have cleaned up live files from object storage', async function () {
+ await checkFilesCleanup(servers[0], videoUUID, [ 720 ])
})
})
- describe('With live transcoding', async function () {
- let videoUUIDPermanent: string
- let videoUUIDNonPermanent: string
+ describe('With live transcoding', function () {
+ const resolutions = [ 720, 480, 360, 240, 144 ]
before(async function () {
await servers[0].config.enableLive({ transcoding: true })
+ })
+
+ describe('Normal replay', function () {
+ let videoUUIDNonPermanent: string
+
+ before(async function () {
+ videoUUIDNonPermanent = await createLive(servers[0], false)
+ })
+
+ it('Should create a live and publish it on object storage', async function () {
+ this.timeout(240000)
+
+ const ffmpegCommand = await servers[0].live.sendRTMPStreamInVideo({ videoId: videoUUIDNonPermanent })
+ await waitUntilLivePublishedOnAllServers(servers, videoUUIDNonPermanent)
+
+ await testVideoResolutions({
+ originServer: servers[0],
+ servers,
+ liveVideoId: videoUUIDNonPermanent,
+ resolutions,
+ transcoded: true,
+ objectStorage: true
+ })
+
+ await stopFfmpeg(ffmpegCommand)
+ })
- videoUUIDPermanent = await createLive(servers[0], true)
- videoUUIDNonPermanent = await createLive(servers[0], false)
+ it('Should have saved the replay on object storage', async function () {
+ this.timeout(220000)
+
+ await waitUntilLiveReplacedByReplayOnAllServers(servers, videoUUIDNonPermanent)
+ await waitJobs(servers)
+
+ await checkFilesExist(servers, videoUUIDNonPermanent, 5)
+ })
+
+ it('Should have cleaned up live files from object storage', async function () {
+ await checkFilesCleanup(servers[0], videoUUIDNonPermanent, resolutions)
+ })
})
- it('Should create a live and save the replay on object storage', async function () {
- this.timeout(240000)
+ describe('Permanent replay', function () {
+ let videoUUIDPermanent: string
+
+ before(async function () {
+ videoUUIDPermanent = await createLive(servers[0], true)
+ })
+
+ it('Should create a live and publish it on object storage', async function () {
+ this.timeout(240000)
+
+ const ffmpegCommand = await servers[0].live.sendRTMPStreamInVideo({ videoId: videoUUIDPermanent })
+ await waitUntilLivePublishedOnAllServers(servers, videoUUIDPermanent)
+
+ await testVideoResolutions({
+ originServer: servers[0],
+ servers,
+ liveVideoId: videoUUIDPermanent,
+ resolutions,
+ transcoded: true,
+ objectStorage: true
+ })
+
+ await stopFfmpeg(ffmpegCommand)
+ })
+
+ it('Should have saved the replay on object storage', async function () {
+ this.timeout(220000)
+
+ await waitUntilLiveWaitingOnAllServers(servers, videoUUIDPermanent)
+ await waitJobs(servers)
+
+ const videoLiveDetails = await servers[0].videos.get({ id: videoUUIDPermanent })
+ const replay = await findExternalSavedVideo(servers[0], videoLiveDetails)
- await streamAndEnd(servers, videoUUIDNonPermanent)
+ await checkFilesExist(servers, replay.uuid, 5)
+ })
- for (const server of servers) {
- const files = await getFiles(server, videoUUIDNonPermanent)
- expect(files).to.have.lengthOf(5)
+ it('Should have cleaned up live files from object storage', async function () {
+ await checkFilesCleanup(servers[0], videoUUIDPermanent, resolutions)
+ })
+ })
+ })
+
+ describe('With object storage base url', function () {
+ const mockObjectStorageProxy = new MockObjectStorageProxy()
+ let baseMockUrl: string
+
+ before(async function () {
+ this.timeout(120000)
+
+ const port = await mockObjectStorageProxy.initialize()
+ baseMockUrl = `http://127.0.0.1:${port}/streaming-playlists`
- await checkFiles(files)
+ await ObjectStorageCommand.createMockBucket('streaming-playlists')
+
+ const config = {
+ object_storage: {
+ enabled: true,
+ endpoint: 'http://' + ObjectStorageCommand.getMockEndpointHost(),
+ region: ObjectStorageCommand.getMockRegion(),
+
+ credentials: ObjectStorageCommand.getMockCredentialsConfig(),
+
+ streaming_playlists: {
+ bucket_name: 'streaming-playlists',
+ prefix: '',
+ base_url: baseMockUrl
+ }
+ }
}
+
+ await servers[0].kill()
+ await servers[0].run(config)
+
+ await servers[0].config.enableLive({ transcoding: true, resolutions: 'min' })
})
- it('Should create a live and save the replay of permanent live on object storage', async function () {
+ it('Should publish a live and replace the base url', async function () {
this.timeout(240000)
- const { videoLiveDetails } = await streamAndEnd(servers, videoUUIDPermanent)
+ const videoUUIDPermanent = await createLive(servers[0], true)
- const replay = await findExternalSavedVideo(servers[0], videoLiveDetails)
+ const ffmpegCommand = await servers[0].live.sendRTMPStreamInVideo({ videoId: videoUUIDPermanent })
+ await waitUntilLivePublishedOnAllServers(servers, videoUUIDPermanent)
- for (const server of servers) {
- const files = await getFiles(server, replay.uuid)
- expect(files).to.have.lengthOf(5)
+ await testVideoResolutions({
+ originServer: servers[0],
+ servers,
+ liveVideoId: videoUUIDPermanent,
+ resolutions: [ 720 ],
+ transcoded: true,
+ objectStorage: true,
+ objectStorageBaseUrl: baseMockUrl
+ })
- await checkFiles(files)
- }
+ await stopFfmpeg(ffmpegCommand)
})
})