From bfbd912886eba17b4aa9a40dcef2fddc685d85bf Mon Sep 17 00:00:00 2001 From: Chocobozzz Date: Wed, 31 Jul 2019 15:57:32 +0200 Subject: Fix broken playlist api --- server/tests/api/check-params/video-playlists.ts | 34 +- server/tests/api/check-params/videos-filter.ts | 26 +- server/tests/api/check-params/videos.ts | 2 +- server/tests/api/videos/video-playlists.ts | 1473 +++++++++++++--------- 4 files changed, 898 insertions(+), 637 deletions(-) (limited to 'server/tests/api') diff --git a/server/tests/api/check-params/video-playlists.ts b/server/tests/api/check-params/video-playlists.ts index 8c5e44bdd..ae5aa287f 100644 --- a/server/tests/api/check-params/video-playlists.ts +++ b/server/tests/api/check-params/video-playlists.ts @@ -37,6 +37,7 @@ describe('Test video playlists API validator', function () { let watchLaterPlaylistId: number let videoId: number let videoId2: number + let playlistElementId: number // --------------------------------------------------------------- @@ -132,18 +133,18 @@ describe('Test video playlists API validator', function () { }) describe('When listing videos of a playlist', function () { - const path = '/api/v1/video-playlists' + const path = '/api/v1/video-playlists/' it('Should fail with a bad start pagination', async function () { - await checkBadStartPagination(server.url, path, server.accessToken) + await checkBadStartPagination(server.url, path + playlistUUID + '/videos', server.accessToken) }) it('Should fail with a bad count pagination', async function () { - await checkBadCountPagination(server.url, path, server.accessToken) + await checkBadCountPagination(server.url, path + playlistUUID + '/videos', server.accessToken) }) - it('Should fail with a bad filter', async function () { - await checkBadSortPagination(server.url, path, server.accessToken) + it('Should success with the correct parameters', async function () { + await makeGetRequest({ url: server.url, path: path + playlistUUID + '/videos', statusCodeExpected: 200 }) }) }) @@ -296,7 +297,7 @@ describe('Test video playlists API validator', function () { token: server.accessToken, playlistId: playlistUUID, elementAttrs: Object.assign({ - videoId: videoId, + videoId, startTimestamp: 2, stopTimestamp: 3 }, elementAttrs) @@ -344,7 +345,8 @@ describe('Test video playlists API validator', function () { it('Succeed with the correct params', async function () { const params = getBase({}, { expectedStatus: 200 }) - await addVideoInPlaylist(params) + const res = await addVideoInPlaylist(params) + playlistElementId = res.body.videoPlaylistElement.id }) it('Should fail if the video was already added in the playlist', async function () { @@ -362,7 +364,7 @@ describe('Test video playlists API validator', function () { startTimestamp: 1, stopTimestamp: 2 }, elementAttrs), - videoId: videoId, + playlistElementId, playlistId: playlistUUID, expectedStatus: 400 }, wrapper) @@ -390,14 +392,14 @@ describe('Test video playlists API validator', function () { } }) - it('Should fail with an unknown or incorrect video id', async function () { + it('Should fail with an unknown or incorrect playlistElement id', async function () { { - const params = getBase({}, { videoId: 'toto' }) + const params = getBase({}, { playlistElementId: 'toto' }) await updateVideoPlaylistElement(params) } { - const params = getBase({}, { videoId: 42, expectedStatus: 404 }) + const params = getBase({}, { playlistElementId: 42, expectedStatus: 404 }) await updateVideoPlaylistElement(params) } }) @@ -415,7 +417,7 @@ describe('Test video playlists API validator', function () { }) it('Should fail with an unknown element', async function () { - const params = getBase({}, { videoId: videoId2, expectedStatus: 404 }) + const params = getBase({}, { playlistElementId: 888, expectedStatus: 404 }) await updateVideoPlaylistElement(params) }) @@ -587,7 +589,7 @@ describe('Test video playlists API validator', function () { return Object.assign({ url: server.url, token: server.accessToken, - videoId: videoId, + playlistElementId, playlistId: playlistUUID, expectedStatus: 400 }, wrapper) @@ -617,18 +619,18 @@ describe('Test video playlists API validator', function () { it('Should fail with an unknown or incorrect video id', async function () { { - const params = getBase({ videoId: 'toto' }) + const params = getBase({ playlistElementId: 'toto' }) await removeVideoFromPlaylist(params) } { - const params = getBase({ videoId: 42, expectedStatus: 404 }) + const params = getBase({ playlistElementId: 42, expectedStatus: 404 }) await removeVideoFromPlaylist(params) } }) it('Should fail with an unknown element', async function () { - const params = getBase({ videoId: videoId2, expectedStatus: 404 }) + const params = getBase({ playlistElementId: 888, expectedStatus: 404 }) await removeVideoFromPlaylist(params) }) diff --git a/server/tests/api/check-params/videos-filter.ts b/server/tests/api/check-params/videos-filter.ts index babef8223..5a5668665 100644 --- a/server/tests/api/check-params/videos-filter.ts +++ b/server/tests/api/check-params/videos-filter.ts @@ -15,13 +15,12 @@ import { import { UserRole } from '../../../../shared/models/users' import { VideoPlaylistPrivacy } from '../../../../shared/models/videos/playlist/video-playlist-privacy.model' -async function testEndpoints (server: ServerInfo, token: string, filter: string, playlistUUID: string, statusCodeExpected: number) { +async function testEndpoints (server: ServerInfo, token: string, filter: string, statusCodeExpected: number) { const paths = [ '/api/v1/video-channels/root_channel/videos', '/api/v1/accounts/root/videos', '/api/v1/videos', - '/api/v1/search/videos', - '/api/v1/video-playlists/' + playlistUUID + '/videos' + '/api/v1/search/videos' ] for (const path of paths) { @@ -70,39 +69,28 @@ describe('Test videos filters', function () { } ) moderatorAccessToken = await userLogin(server, moderator) - - const res = await createVideoPlaylist({ - url: server.url, - token: server.accessToken, - playlistAttrs: { - displayName: 'super playlist', - privacy: VideoPlaylistPrivacy.PUBLIC, - videoChannelId: server.videoChannel.id - } - }) - playlistUUID = res.body.videoPlaylist.uuid }) describe('When setting a video filter', function () { it('Should fail with a bad filter', async function () { - await testEndpoints(server, server.accessToken, 'bad-filter', playlistUUID, 400) + await testEndpoints(server, server.accessToken, 'bad-filter', 400) }) it('Should succeed with a good filter', async function () { - await testEndpoints(server, server.accessToken,'local', playlistUUID, 200) + await testEndpoints(server, server.accessToken,'local', 200) }) it('Should fail to list all-local with a simple user', async function () { - await testEndpoints(server, userAccessToken, 'all-local', playlistUUID, 401) + await testEndpoints(server, userAccessToken, 'all-local', 401) }) it('Should succeed to list all-local with a moderator', async function () { - await testEndpoints(server, moderatorAccessToken, 'all-local', playlistUUID, 200) + await testEndpoints(server, moderatorAccessToken, 'all-local', 200) }) it('Should succeed to list all-local with an admin', async function () { - await testEndpoints(server, server.accessToken, 'all-local', playlistUUID, 200) + await testEndpoints(server, server.accessToken, 'all-local', 200) }) // Because we cannot authenticate the user on the RSS endpoint diff --git a/server/tests/api/check-params/videos.ts b/server/tests/api/check-params/videos.ts index 51e592a15..fa6d6f622 100644 --- a/server/tests/api/check-params/videos.ts +++ b/server/tests/api/check-params/videos.ts @@ -62,7 +62,7 @@ describe('Test videos API validator', function () { } }) - describe('When listing a video', function () { + describe('When listing videos', function () { it('Should fail with a bad start pagination', async function () { await checkBadStartPagination(server.url, path) }) diff --git a/server/tests/api/videos/video-playlists.ts b/server/tests/api/videos/video-playlists.ts index f82c8cbce..7d5e3914b 100644 --- a/server/tests/api/videos/video-playlists.ts +++ b/server/tests/api/videos/video-playlists.ts @@ -5,6 +5,7 @@ import 'mocha' import { addVideoChannel, addVideoInPlaylist, + addVideoToBlacklist, checkPlaylistFilesWereRemoved, cleanupTests, createUser, @@ -14,6 +15,8 @@ import { doubleFollow, doVideosExistInMyPlaylist, flushAndRunMultipleServers, + generateUserAccessToken, + getAccessToken, getAccountPlaylistsList, getAccountPlaylistsListWithToken, getMyUserInformation, @@ -24,6 +27,7 @@ import { getVideoPlaylistsList, getVideoPlaylistWithToken, removeUser, + removeVideoFromBlacklist, removeVideoFromPlaylist, reorderVideosPlaylist, ServerInfo, @@ -31,23 +35,58 @@ import { setDefaultVideoChannel, testImage, unfollow, + updateVideo, updateVideoPlaylist, updateVideoPlaylistElement, uploadVideo, uploadVideoAndGetId, userLogin, - waitJobs, - generateUserAccessToken + waitJobs } from '../../../../shared/extra-utils' import { VideoPlaylistPrivacy } from '../../../../shared/models/videos/playlist/video-playlist-privacy.model' import { VideoPlaylist } from '../../../../shared/models/videos/playlist/video-playlist.model' -import { Video } from '../../../../shared/models/videos' +import { VideoPrivacy } from '../../../../shared/models/videos' import { VideoPlaylistType } from '../../../../shared/models/videos/playlist/video-playlist-type.model' import { VideoExistInPlaylist } from '../../../../shared/models/videos/playlist/video-exist-in-playlist.model' import { User } from '../../../../shared/models/users' +import { VideoPlaylistElement, VideoPlaylistElementType } from '../../../../shared/models/videos/playlist/video-playlist-element.model' +import { + addAccountToAccountBlocklist, + addAccountToServerBlocklist, + addServerToAccountBlocklist, + addServerToServerBlocklist, + removeAccountFromAccountBlocklist, + removeAccountFromServerBlocklist, + removeServerFromAccountBlocklist, + removeServerFromServerBlocklist +} from '../../../../shared/extra-utils/users/blocklist' const expect = chai.expect +async function checkPlaylistElementType ( + servers: ServerInfo[], + playlistId: string, + type: VideoPlaylistElementType, + position: number, + name: string, + total: number +) { + for (const server of servers) { + const res = await getPlaylistVideos(server.url, server.accessToken, playlistId, 0, 10) + expect(res.body.total).to.equal(total) + + const videoElement: VideoPlaylistElement = res.body.data.find((e: VideoPlaylistElement) => e.position === position) + expect(videoElement.type).to.equal(type, 'On server ' + server.url) + + if (type === VideoPlaylistElementType.REGULAR) { + expect(videoElement.video).to.not.be.null + expect(videoElement.video.name).to.equal(name) + } else { + expect(videoElement.video).to.be.null + } + } +} + describe('Test video playlists', function () { let servers: ServerInfo[] = [] @@ -57,9 +96,16 @@ describe('Test video playlists', function () { let playlistServer1Id: number let playlistServer1UUID: string + let playlistServer1UUID2: string + + let playlistElementServer1Video4: number + let playlistElementServer1Video5: number + let playlistElementNSFW: number let nsfwVideoServer1: number + let userAccessTokenServer1: string + before(async function () { this.timeout(120000) @@ -97,814 +143,1039 @@ describe('Test video playlists', function () { nsfwVideoServer1 = (await uploadVideoAndGetId({ server: servers[ 0 ], videoName: 'NSFW video', nsfw: true })).id + { + await createUser({ + url: servers[ 0 ].url, + accessToken: servers[ 0 ].accessToken, + username: 'user1', + password: 'password' + }) + userAccessTokenServer1 = await getAccessToken(servers[0].url, 'user1', 'password') + } + await waitJobs(servers) }) - it('Should list video playlist privacies', async function () { - const res = await getVideoPlaylistPrivacies(servers[0].url) + describe('Get default playlists', function () { + it('Should list video playlist privacies', async function () { + const res = await getVideoPlaylistPrivacies(servers[ 0 ].url) - const privacies = res.body - expect(Object.keys(privacies)).to.have.length.at.least(3) + const privacies = res.body + expect(Object.keys(privacies)).to.have.length.at.least(3) - expect(privacies[3]).to.equal('Private') - }) + expect(privacies[ 3 ]).to.equal('Private') + }) - it('Should list watch later playlist', async function () { - const url = servers[ 0 ].url - const accessToken = servers[ 0 ].accessToken + it('Should list watch later playlist', async function () { + const url = servers[ 0 ].url + const accessToken = servers[ 0 ].accessToken - { - const res = await getAccountPlaylistsListWithToken(url, accessToken, 'root', 0, 5, VideoPlaylistType.WATCH_LATER) + { + const res = await getAccountPlaylistsListWithToken(url, accessToken, 'root', 0, 5, VideoPlaylistType.WATCH_LATER) + + expect(res.body.total).to.equal(1) + expect(res.body.data).to.have.lengthOf(1) + + const playlist: VideoPlaylist = res.body.data[ 0 ] + expect(playlist.displayName).to.equal('Watch later') + expect(playlist.type.id).to.equal(VideoPlaylistType.WATCH_LATER) + expect(playlist.type.label).to.equal('Watch later') + } + + { + const res = await getAccountPlaylistsListWithToken(url, accessToken, 'root', 0, 5, VideoPlaylistType.REGULAR) + + expect(res.body.total).to.equal(0) + expect(res.body.data).to.have.lengthOf(0) + } + + { + const res = await getAccountPlaylistsList(url, 'root', 0, 5) + expect(res.body.total).to.equal(0) + expect(res.body.data).to.have.lengthOf(0) + } + }) + + it('Should get private playlist for a classic user', async function () { + const token = await generateUserAccessToken(servers[ 0 ], 'toto') + + const res = await getAccountPlaylistsListWithToken(servers[ 0 ].url, token, 'toto', 0, 5) expect(res.body.total).to.equal(1) expect(res.body.data).to.have.lengthOf(1) - const playlist: VideoPlaylist = res.body.data[ 0 ] - expect(playlist.displayName).to.equal('Watch later') - expect(playlist.type.id).to.equal(VideoPlaylistType.WATCH_LATER) - expect(playlist.type.label).to.equal('Watch later') - } + const playlistId = res.body.data[ 0 ].id + await getPlaylistVideos(servers[ 0 ].url, token, playlistId, 0, 5) + }) + }) - { - const res = await getAccountPlaylistsListWithToken(url, accessToken, 'root', 0, 5, VideoPlaylistType.REGULAR) + describe('Create and federate playlists', function () { - expect(res.body.total).to.equal(0) - expect(res.body.data).to.have.lengthOf(0) - } + it('Should create a playlist on server 1 and have the playlist on server 2 and 3', async function () { + this.timeout(30000) - { - const res = await getAccountPlaylistsList(url, 'root', 0, 5) - expect(res.body.total).to.equal(0) - expect(res.body.data).to.have.lengthOf(0) - } - }) + await createVideoPlaylist({ + url: servers[ 0 ].url, + token: servers[ 0 ].accessToken, + playlistAttrs: { + displayName: 'my super playlist', + privacy: VideoPlaylistPrivacy.PUBLIC, + description: 'my super description', + thumbnailfile: 'thumbnail.jpg', + videoChannelId: servers[ 0 ].videoChannel.id + } + }) + + await waitJobs(servers) + + for (const server of servers) { + const res = await getVideoPlaylistsList(server.url, 0, 5) + expect(res.body.total).to.equal(1) + expect(res.body.data).to.have.lengthOf(1) + + const playlistFromList = res.body.data[ 0 ] as VideoPlaylist + + const res2 = await getVideoPlaylist(server.url, playlistFromList.uuid) + const playlistFromGet = res2.body + + for (const playlist of [ playlistFromGet, playlistFromList ]) { + expect(playlist.id).to.be.a('number') + expect(playlist.uuid).to.be.a('string') + + expect(playlist.isLocal).to.equal(server.serverNumber === 1) + + expect(playlist.displayName).to.equal('my super playlist') + expect(playlist.description).to.equal('my super description') + expect(playlist.privacy.id).to.equal(VideoPlaylistPrivacy.PUBLIC) + expect(playlist.privacy.label).to.equal('Public') + expect(playlist.type.id).to.equal(VideoPlaylistType.REGULAR) + expect(playlist.type.label).to.equal('Regular') + + expect(playlist.videosLength).to.equal(0) - it('Should get private playlist for a classic user', async function () { - const token = await generateUserAccessToken(servers[0], 'toto') + expect(playlist.ownerAccount.name).to.equal('root') + expect(playlist.ownerAccount.displayName).to.equal('root') + expect(playlist.videoChannel.name).to.equal('root_channel') + expect(playlist.videoChannel.displayName).to.equal('Main root channel') + } + } + }) - const res = await getAccountPlaylistsListWithToken(servers[0].url, token, 'toto', 0, 5) + it('Should create a playlist on server 2 and have the playlist on server 1 but not on server 3', async function () { + this.timeout(30000) + + { + const res = await createVideoPlaylist({ + url: servers[ 1 ].url, + token: servers[ 1 ].accessToken, + playlistAttrs: { + displayName: 'playlist 2', + privacy: VideoPlaylistPrivacy.PUBLIC, + videoChannelId: servers[ 1 ].videoChannel.id + } + }) + playlistServer2Id1 = res.body.videoPlaylist.id + } - expect(res.body.total).to.equal(1) - expect(res.body.data).to.have.lengthOf(1) + { + const res = await createVideoPlaylist({ + url: servers[ 1 ].url, + token: servers[ 1 ].accessToken, + playlistAttrs: { + displayName: 'playlist 3', + privacy: VideoPlaylistPrivacy.PUBLIC, + thumbnailfile: 'thumbnail.jpg', + videoChannelId: servers[ 1 ].videoChannel.id + } + }) + + playlistServer2Id2 = res.body.videoPlaylist.id + playlistServer2UUID2 = res.body.videoPlaylist.uuid + } + + for (let id of [ playlistServer2Id1, playlistServer2Id2 ]) { + await addVideoInPlaylist({ + url: servers[ 1 ].url, + token: servers[ 1 ].accessToken, + playlistId: id, + elementAttrs: { videoId: servers[ 1 ].videos[ 0 ].id, startTimestamp: 1, stopTimestamp: 2 } + }) + await addVideoInPlaylist({ + url: servers[ 1 ].url, + token: servers[ 1 ].accessToken, + playlistId: id, + elementAttrs: { videoId: servers[ 1 ].videos[ 1 ].id } + }) + } + + await waitJobs(servers) + + for (const server of [ servers[ 0 ], servers[ 1 ] ]) { + const res = await getVideoPlaylistsList(server.url, 0, 5) + + const playlist2 = res.body.data.find(p => p.displayName === 'playlist 2') + expect(playlist2).to.not.be.undefined + await testImage(server.url, 'thumbnail-playlist', playlist2.thumbnailPath) + + const playlist3 = res.body.data.find(p => p.displayName === 'playlist 3') + expect(playlist3).to.not.be.undefined + await testImage(server.url, 'thumbnail', playlist3.thumbnailPath) + } - const playlistId = res.body.data[0].id - await getPlaylistVideos(servers[0].url, token, playlistId, 0, 5) + const res = await getVideoPlaylistsList(servers[ 2 ].url, 0, 5) + expect(res.body.data.find(p => p.displayName === 'playlist 2')).to.be.undefined + expect(res.body.data.find(p => p.displayName === 'playlist 3')).to.be.undefined + }) + + it('Should have the playlist on server 3 after a new follow', async function () { + this.timeout(30000) + + // Server 2 and server 3 follow each other + await doubleFollow(servers[ 1 ], servers[ 2 ]) + + const res = await getVideoPlaylistsList(servers[ 2 ].url, 0, 5) + + const playlist2 = res.body.data.find(p => p.displayName === 'playlist 2') + expect(playlist2).to.not.be.undefined + await testImage(servers[ 2 ].url, 'thumbnail-playlist', playlist2.thumbnailPath) + + expect(res.body.data.find(p => p.displayName === 'playlist 3')).to.not.be.undefined + }) }) - it('Should create a playlist on server 1 and have the playlist on server 2 and 3', async function () { - this.timeout(30000) + describe('List playlists', function () { + it('Should correctly list the playlists', async function () { + this.timeout(30000) + + { + const res = await getVideoPlaylistsList(servers[ 2 ].url, 1, 2, 'createdAt') - await createVideoPlaylist({ - url: servers[0].url, - token: servers[0].accessToken, - playlistAttrs: { - displayName: 'my super playlist', - privacy: VideoPlaylistPrivacy.PUBLIC, - description: 'my super description', - thumbnailfile: 'thumbnail.jpg', - videoChannelId: servers[0].videoChannel.id + expect(res.body.total).to.equal(3) + + const data: VideoPlaylist[] = res.body.data + expect(data).to.have.lengthOf(2) + expect(data[ 0 ].displayName).to.equal('playlist 2') + expect(data[ 1 ].displayName).to.equal('playlist 3') + } + + { + const res = await getVideoPlaylistsList(servers[ 2 ].url, 1, 2, '-createdAt') + + expect(res.body.total).to.equal(3) + + const data: VideoPlaylist[] = res.body.data + expect(data).to.have.lengthOf(2) + expect(data[ 0 ].displayName).to.equal('playlist 2') + expect(data[ 1 ].displayName).to.equal('my super playlist') } }) - await waitJobs(servers) + it('Should list video channel playlists', async function () { + this.timeout(30000) - for (const server of servers) { - const res = await getVideoPlaylistsList(server.url, 0, 5) - expect(res.body.total).to.equal(1) - expect(res.body.data).to.have.lengthOf(1) + { + const res = await getVideoChannelPlaylistsList(servers[ 0 ].url, 'root_channel', 0, 2, '-createdAt') - const playlistFromList = res.body.data[0] as VideoPlaylist + expect(res.body.total).to.equal(1) - const res2 = await getVideoPlaylist(server.url, playlistFromList.uuid) - const playlistFromGet = res2.body + const data: VideoPlaylist[] = res.body.data + expect(data).to.have.lengthOf(1) + expect(data[ 0 ].displayName).to.equal('my super playlist') + } + }) - for (const playlist of [ playlistFromGet, playlistFromList ]) { - expect(playlist.id).to.be.a('number') - expect(playlist.uuid).to.be.a('string') + it('Should list account playlists', async function () { + this.timeout(30000) - expect(playlist.isLocal).to.equal(server.serverNumber === 1) + { + const res = await getAccountPlaylistsList(servers[ 1 ].url, 'root', 1, 2, '-createdAt') - expect(playlist.displayName).to.equal('my super playlist') - expect(playlist.description).to.equal('my super description') - expect(playlist.privacy.id).to.equal(VideoPlaylistPrivacy.PUBLIC) - expect(playlist.privacy.label).to.equal('Public') - expect(playlist.type.id).to.equal(VideoPlaylistType.REGULAR) - expect(playlist.type.label).to.equal('Regular') + expect(res.body.total).to.equal(2) + + const data: VideoPlaylist[] = res.body.data + expect(data).to.have.lengthOf(1) + expect(data[ 0 ].displayName).to.equal('playlist 2') + } - expect(playlist.videosLength).to.equal(0) + { + const res = await getAccountPlaylistsList(servers[ 1 ].url, 'root', 1, 2, 'createdAt') - expect(playlist.ownerAccount.name).to.equal('root') - expect(playlist.ownerAccount.displayName).to.equal('root') - expect(playlist.videoChannel.name).to.equal('root_channel') - expect(playlist.videoChannel.displayName).to.equal('Main root channel') + expect(res.body.total).to.equal(2) + + const data: VideoPlaylist[] = res.body.data + expect(data).to.have.lengthOf(1) + expect(data[ 0 ].displayName).to.equal('playlist 3') } - } - }) + }) - it('Should create a playlist on server 2 and have the playlist on server 1 but not on server 3', async function () { - this.timeout(30000) + it('Should not list unlisted or private playlists', async function () { + this.timeout(30000) - { - const res = await createVideoPlaylist({ - url: servers[1].url, - token: servers[1].accessToken, + await createVideoPlaylist({ + url: servers[ 1 ].url, + token: servers[ 1 ].accessToken, playlistAttrs: { - displayName: 'playlist 2', - privacy: VideoPlaylistPrivacy.PUBLIC, - videoChannelId: servers[1].videoChannel.id + displayName: 'playlist unlisted', + privacy: VideoPlaylistPrivacy.UNLISTED } }) - playlistServer2Id1 = res.body.videoPlaylist.id - } - { - const res = await createVideoPlaylist({ + await createVideoPlaylist({ url: servers[ 1 ].url, token: servers[ 1 ].accessToken, playlistAttrs: { - displayName: 'playlist 3', - privacy: VideoPlaylistPrivacy.PUBLIC, - thumbnailfile: 'thumbnail.jpg', - videoChannelId: servers[1].videoChannel.id + displayName: 'playlist private', + privacy: VideoPlaylistPrivacy.PRIVATE } }) - playlistServer2Id2 = res.body.videoPlaylist.id - playlistServer2UUID2 = res.body.videoPlaylist.uuid - } + await waitJobs(servers) - for (let id of [ playlistServer2Id1, playlistServer2Id2 ]) { - await addVideoInPlaylist({ - url: servers[ 1 ].url, - token: servers[ 1 ].accessToken, - playlistId: id, - elementAttrs: { videoId: servers[ 1 ].videos[ 0 ].id, startTimestamp: 1, stopTimestamp: 2 } - }) - await addVideoInPlaylist({ - url: servers[ 1 ].url, - token: servers[ 1 ].accessToken, - playlistId: id, - elementAttrs: { videoId: servers[ 1 ].videos[ 1 ].id } - }) - } + for (const server of servers) { + const results = [ + await getAccountPlaylistsList(server.url, 'root@localhost:' + servers[ 1 ].port, 0, 5, '-createdAt'), + await getVideoPlaylistsList(server.url, 0, 2, '-createdAt') + ] + + expect(results[ 0 ].body.total).to.equal(2) + expect(results[ 1 ].body.total).to.equal(3) + + for (const res of results) { + const data: VideoPlaylist[] = res.body.data + expect(data).to.have.lengthOf(2) + expect(data[ 0 ].displayName).to.equal('playlist 3') + expect(data[ 1 ].displayName).to.equal('playlist 2') + } + } + }) + }) - await waitJobs(servers) + describe('Update playlists', function () { - for (const server of [ servers[0], servers[1] ]) { - const res = await getVideoPlaylistsList(server.url, 0, 5) + it('Should update a playlist', async function () { + this.timeout(30000) - const playlist2 = res.body.data.find(p => p.displayName === 'playlist 2') - expect(playlist2).to.not.be.undefined - await testImage(server.url, 'thumbnail-playlist', playlist2.thumbnailPath) + await updateVideoPlaylist({ + url: servers[1].url, + token: servers[1].accessToken, + playlistAttrs: { + displayName: 'playlist 3 updated', + description: 'description updated', + privacy: VideoPlaylistPrivacy.UNLISTED, + thumbnailfile: 'thumbnail.jpg', + videoChannelId: servers[1].videoChannel.id + }, + playlistId: playlistServer2Id2 + }) - const playlist3 = res.body.data.find(p => p.displayName === 'playlist 3') - expect(playlist3).to.not.be.undefined - await testImage(server.url, 'thumbnail', playlist3.thumbnailPath) - } + await waitJobs(servers) - const res = await getVideoPlaylistsList(servers[2].url, 0, 5) - expect(res.body.data.find(p => p.displayName === 'playlist 2')).to.be.undefined - expect(res.body.data.find(p => p.displayName === 'playlist 3')).to.be.undefined - }) + for (const server of servers) { + const res = await getVideoPlaylist(server.url, playlistServer2UUID2) + const playlist: VideoPlaylist = res.body - it('Should have the playlist on server 3 after a new follow', async function () { - this.timeout(30000) + expect(playlist.displayName).to.equal('playlist 3 updated') + expect(playlist.description).to.equal('description updated') - // Server 2 and server 3 follow each other - await doubleFollow(servers[1], servers[2]) + expect(playlist.privacy.id).to.equal(VideoPlaylistPrivacy.UNLISTED) + expect(playlist.privacy.label).to.equal('Unlisted') - const res = await getVideoPlaylistsList(servers[2].url, 0, 5) + expect(playlist.type.id).to.equal(VideoPlaylistType.REGULAR) + expect(playlist.type.label).to.equal('Regular') - const playlist2 = res.body.data.find(p => p.displayName === 'playlist 2') - expect(playlist2).to.not.be.undefined - await testImage(servers[2].url, 'thumbnail-playlist', playlist2.thumbnailPath) + expect(playlist.videosLength).to.equal(2) - expect(res.body.data.find(p => p.displayName === 'playlist 3')).to.not.be.undefined + expect(playlist.ownerAccount.name).to.equal('root') + expect(playlist.ownerAccount.displayName).to.equal('root') + expect(playlist.videoChannel.name).to.equal('root_channel') + expect(playlist.videoChannel.displayName).to.equal('Main root channel') + } + }) }) - it('Should correctly list the playlists', async function () { - this.timeout(30000) + describe('Element timestamps', function () { - { - const res = await getVideoPlaylistsList(servers[ 2 ].url, 1, 2, 'createdAt') + it('Should create a playlist containing different startTimestamp/endTimestamp videos', async function () { + this.timeout(30000) - expect(res.body.total).to.equal(3) + const addVideo = (elementAttrs: any) => { + return addVideoInPlaylist({ url: servers[ 0 ].url, token: servers[ 0 ].accessToken, playlistId: playlistServer1Id, elementAttrs }) + } - const data: VideoPlaylist[] = res.body.data - expect(data).to.have.lengthOf(2) - expect(data[ 0 ].displayName).to.equal('playlist 2') - expect(data[ 1 ].displayName).to.equal('playlist 3') - } + const res = await createVideoPlaylist({ + url: servers[ 0 ].url, + token: servers[ 0 ].accessToken, + playlistAttrs: { + displayName: 'playlist 4', + privacy: VideoPlaylistPrivacy.PUBLIC, + videoChannelId: servers[ 0 ].videoChannel.id + } + }) - { - const res = await getVideoPlaylistsList(servers[ 2 ].url, 1, 2, '-createdAt') + playlistServer1Id = res.body.videoPlaylist.id + playlistServer1UUID = res.body.videoPlaylist.uuid - expect(res.body.total).to.equal(3) + await addVideo({ videoId: servers[ 0 ].videos[ 0 ].uuid, startTimestamp: 15, stopTimestamp: 28 }) + await addVideo({ videoId: servers[ 2 ].videos[ 1 ].uuid, startTimestamp: 35 }) + await addVideo({ videoId: servers[ 2 ].videos[ 2 ].uuid }) + { + const res = await addVideo({ videoId: servers[ 0 ].videos[ 3 ].uuid, stopTimestamp: 35 }) + playlistElementServer1Video4 = res.body.videoPlaylistElement.id + } - const data: VideoPlaylist[] = res.body.data - expect(data).to.have.lengthOf(2) - expect(data[ 0 ].displayName).to.equal('playlist 2') - expect(data[ 1 ].displayName).to.equal('my super playlist') - } - }) + { + const res = await addVideo({ videoId: servers[ 0 ].videos[ 4 ].uuid, startTimestamp: 45, stopTimestamp: 60 }) + playlistElementServer1Video5 = res.body.videoPlaylistElement.id + } - it('Should list video channel playlists', async function () { - this.timeout(30000) + { + const res = await addVideo({ videoId: nsfwVideoServer1, startTimestamp: 5 }) + playlistElementNSFW = res.body.videoPlaylistElement.id + } - { - const res = await getVideoChannelPlaylistsList(servers[ 0 ].url, 'root_channel', 0, 2, '-createdAt') + await waitJobs(servers) + }) - expect(res.body.total).to.equal(1) + it('Should correctly list playlist videos', async function () { + this.timeout(30000) - const data: VideoPlaylist[] = res.body.data - expect(data).to.have.lengthOf(1) - expect(data[ 0 ].displayName).to.equal('my super playlist') - } - }) + for (const server of servers) { + const res = await getPlaylistVideos(server.url, server.accessToken, playlistServer1UUID, 0, 10) - it('Should list account playlists', async function () { - this.timeout(30000) + expect(res.body.total).to.equal(6) - { - const res = await getAccountPlaylistsList(servers[ 1 ].url, 'root', 1, 2, '-createdAt') + const videoElements: VideoPlaylistElement[] = res.body.data + expect(videoElements).to.have.lengthOf(6) - expect(res.body.total).to.equal(2) + expect(videoElements[ 0 ].video.name).to.equal('video 0 server 1') + expect(videoElements[ 0 ].position).to.equal(1) + expect(videoElements[ 0 ].startTimestamp).to.equal(15) + expect(videoElements[ 0 ].stopTimestamp).to.equal(28) - const data: VideoPlaylist[] = res.body.data - expect(data).to.have.lengthOf(1) - expect(data[ 0 ].displayName).to.equal('playlist 2') - } + expect(videoElements[ 1 ].video.name).to.equal('video 1 server 3') + expect(videoElements[ 1 ].position).to.equal(2) + expect(videoElements[ 1 ].startTimestamp).to.equal(35) + expect(videoElements[ 1 ].stopTimestamp).to.be.null - { - const res = await getAccountPlaylistsList(servers[ 1 ].url, 'root', 1, 2, 'createdAt') + expect(videoElements[ 2 ].video.name).to.equal('video 2 server 3') + expect(videoElements[ 2 ].position).to.equal(3) + expect(videoElements[ 2 ].startTimestamp).to.be.null + expect(videoElements[ 2 ].stopTimestamp).to.be.null - expect(res.body.total).to.equal(2) + expect(videoElements[ 3 ].video.name).to.equal('video 3 server 1') + expect(videoElements[ 3 ].position).to.equal(4) + expect(videoElements[ 3 ].startTimestamp).to.be.null + expect(videoElements[ 3 ].stopTimestamp).to.equal(35) - const data: VideoPlaylist[] = res.body.data - expect(data).to.have.lengthOf(1) - expect(data[ 0 ].displayName).to.equal('playlist 3') - } - }) + expect(videoElements[ 4 ].video.name).to.equal('video 4 server 1') + expect(videoElements[ 4 ].position).to.equal(5) + expect(videoElements[ 4 ].startTimestamp).to.equal(45) + expect(videoElements[ 4 ].stopTimestamp).to.equal(60) - it('Should not list unlisted or private playlists', async function () { - this.timeout(30000) + expect(videoElements[ 5 ].video.name).to.equal('NSFW video') + expect(videoElements[ 5 ].position).to.equal(6) + expect(videoElements[ 5 ].startTimestamp).to.equal(5) + expect(videoElements[ 5 ].stopTimestamp).to.be.null - await createVideoPlaylist({ - url: servers[ 1 ].url, - token: servers[ 1 ].accessToken, - playlistAttrs: { - displayName: 'playlist unlisted', - privacy: VideoPlaylistPrivacy.UNLISTED + const res3 = await getPlaylistVideos(server.url, server.accessToken, playlistServer1UUID, 0, 2) + expect(res3.body.data).to.have.lengthOf(2) } }) + }) - await createVideoPlaylist({ - url: servers[ 1 ].url, - token: servers[ 1 ].accessToken, - playlistAttrs: { - displayName: 'playlist private', - privacy: VideoPlaylistPrivacy.PRIVATE - } - }) + describe('Element type', function () { + let groupUser1: ServerInfo[] + let groupWithoutToken1: ServerInfo[] + let group1: ServerInfo[] + let group2: ServerInfo[] - await waitJobs(servers) + let video1: string + let video2: string + let video3: string - for (const server of servers) { - const results = [ - await getAccountPlaylistsList(server.url, 'root@localhost:' + servers[1].port, 0, 5, '-createdAt'), - await getVideoPlaylistsList(server.url, 0, 2, '-createdAt') - ] + before(async function () { + this.timeout(30000) - expect(results[0].body.total).to.equal(2) - expect(results[1].body.total).to.equal(3) + groupUser1 = [ Object.assign({}, servers[ 0 ], { accessToken: userAccessTokenServer1 }) ] + groupWithoutToken1 = [ Object.assign({}, servers[ 0 ], { accessToken: undefined }) ] + group1 = [ servers[ 0 ] ] + group2 = [ servers[ 1 ], servers[ 2 ] ] - for (const res of results) { - const data: VideoPlaylist[] = res.body.data - expect(data).to.have.lengthOf(2) - expect(data[ 0 ].displayName).to.equal('playlist 3') - expect(data[ 1 ].displayName).to.equal('playlist 2') - } - } - }) + const res = await createVideoPlaylist({ + url: servers[ 0 ].url, + token: userAccessTokenServer1, + playlistAttrs: { + displayName: 'playlist 56', + privacy: VideoPlaylistPrivacy.PUBLIC, + videoChannelId: servers[ 0 ].videoChannel.id + } + }) - it('Should update a playlist', async function () { - this.timeout(30000) - - await updateVideoPlaylist({ - url: servers[1].url, - token: servers[1].accessToken, - playlistAttrs: { - displayName: 'playlist 3 updated', - description: 'description updated', - privacy: VideoPlaylistPrivacy.UNLISTED, - thumbnailfile: 'thumbnail.jpg', - videoChannelId: servers[1].videoChannel.id - }, - playlistId: playlistServer2Id2 - }) + const playlistServer1Id2 = res.body.videoPlaylist.id + playlistServer1UUID2 = res.body.videoPlaylist.uuid - await waitJobs(servers) + const addVideo = (elementAttrs: any) => { + return addVideoInPlaylist({ url: servers[ 0 ].url, token: userAccessTokenServer1, playlistId: playlistServer1Id2, elementAttrs }) + } - for (const server of servers) { - const res = await getVideoPlaylist(server.url, playlistServer2UUID2) - const playlist: VideoPlaylist = res.body + video1 = (await uploadVideoAndGetId({ server: servers[0], videoName: 'video 89', token: userAccessTokenServer1 })).uuid + video2 = (await uploadVideoAndGetId({ server: servers[1], videoName: 'video 90' })).uuid + video3 = (await uploadVideoAndGetId({ server: servers[0], videoName: 'video 91', nsfw: true })).uuid - expect(playlist.displayName).to.equal('playlist 3 updated') - expect(playlist.description).to.equal('description updated') + await addVideo({ videoId: video1, startTimestamp: 15, stopTimestamp: 28 }) + await addVideo({ videoId: video2, startTimestamp: 35 }) + await addVideo({ videoId: video3 }) - expect(playlist.privacy.id).to.equal(VideoPlaylistPrivacy.UNLISTED) - expect(playlist.privacy.label).to.equal('Unlisted') + await waitJobs(servers) + }) - expect(playlist.type.id).to.equal(VideoPlaylistType.REGULAR) - expect(playlist.type.label).to.equal('Regular') + it('Should update the element type if the video is private', async function () { + this.timeout(20000) - expect(playlist.videosLength).to.equal(2) + const name = 'video 89' + const position = 1 - expect(playlist.ownerAccount.name).to.equal('root') - expect(playlist.ownerAccount.displayName).to.equal('root') - expect(playlist.videoChannel.name).to.equal('root_channel') - expect(playlist.videoChannel.displayName).to.equal('Main root channel') - } - }) + { + await updateVideo(servers[ 0 ].url, servers[ 0 ].accessToken, video1, { privacy: VideoPrivacy.PRIVATE }) + await waitJobs(servers) - it('Should create a playlist containing different startTimestamp/endTimestamp videos', async function () { - this.timeout(30000) + await checkPlaylistElementType(groupUser1, playlistServer1UUID2, VideoPlaylistElementType.REGULAR, position, name, 3) + await checkPlaylistElementType(groupWithoutToken1, playlistServer1UUID2, VideoPlaylistElementType.PRIVATE, position, name, 3) + await checkPlaylistElementType(group1, playlistServer1UUID2, VideoPlaylistElementType.PRIVATE, position, name, 3) + await checkPlaylistElementType(group2, playlistServer1UUID2, VideoPlaylistElementType.DELETED, position, name, 3) + } - const addVideo = (elementAttrs: any) => { - return addVideoInPlaylist({ url: servers[0].url, token: servers[0].accessToken, playlistId: playlistServer1Id, elementAttrs }) - } + { + await updateVideo(servers[ 0 ].url, servers[ 0 ].accessToken, video1, { privacy: VideoPrivacy.PUBLIC }) + await waitJobs(servers) - const res = await createVideoPlaylist({ - url: servers[ 0 ].url, - token: servers[ 0 ].accessToken, - playlistAttrs: { - displayName: 'playlist 4', - privacy: VideoPlaylistPrivacy.PUBLIC, - videoChannelId: servers[0].videoChannel.id + await checkPlaylistElementType(groupUser1, playlistServer1UUID2, VideoPlaylistElementType.REGULAR, position, name, 3) + await checkPlaylistElementType(groupWithoutToken1, playlistServer1UUID2, VideoPlaylistElementType.REGULAR, position, name, 3) + await checkPlaylistElementType(group1, playlistServer1UUID2, VideoPlaylistElementType.REGULAR, position, name, 3) + // We deleted the video, so even if we recreated it, the old entry is still deleted + await checkPlaylistElementType(group2, playlistServer1UUID2, VideoPlaylistElementType.DELETED, position, name, 3) } }) - playlistServer1Id = res.body.videoPlaylist.id - playlistServer1UUID = res.body.videoPlaylist.uuid + it('Should update the element type if the video is blacklisted', async function () { + this.timeout(20000) - await addVideo({ videoId: servers[0].videos[0].uuid, startTimestamp: 15, stopTimestamp: 28 }) - await addVideo({ videoId: servers[2].videos[1].uuid, startTimestamp: 35 }) - await addVideo({ videoId: servers[2].videos[2].uuid }) - await addVideo({ videoId: servers[0].videos[3].uuid, stopTimestamp: 35 }) - await addVideo({ videoId: servers[0].videos[4].uuid, startTimestamp: 45, stopTimestamp: 60 }) - await addVideo({ videoId: nsfwVideoServer1, startTimestamp: 5 }) + const name = 'video 89' + const position = 1 - await waitJobs(servers) - }) + { + await addVideoToBlacklist(servers[ 0 ].url, servers[ 0 ].accessToken, video1, 'reason', true) + await waitJobs(servers) - it('Should correctly list playlist videos', async function () { - this.timeout(30000) + await checkPlaylistElementType(groupUser1, playlistServer1UUID2, VideoPlaylistElementType.REGULAR, position, name, 3) + await checkPlaylistElementType(groupWithoutToken1, playlistServer1UUID2, VideoPlaylistElementType.UNAVAILABLE, position, name, 3) + await checkPlaylistElementType(group1, playlistServer1UUID2, VideoPlaylistElementType.UNAVAILABLE, position, name, 3) + await checkPlaylistElementType(group2, playlistServer1UUID2, VideoPlaylistElementType.DELETED, position, name, 3) + } - for (const server of servers) { - const res = await getPlaylistVideos(server.url, server.accessToken, playlistServer1UUID, 0, 10) + { + await removeVideoFromBlacklist(servers[ 0 ].url, servers[ 0 ].accessToken, video1) + await waitJobs(servers) - expect(res.body.total).to.equal(6) + await checkPlaylistElementType(groupUser1, playlistServer1UUID2, VideoPlaylistElementType.REGULAR, position, name, 3) + await checkPlaylistElementType(groupWithoutToken1, playlistServer1UUID2, VideoPlaylistElementType.REGULAR, position, name, 3) + await checkPlaylistElementType(group1, playlistServer1UUID2, VideoPlaylistElementType.REGULAR, position, name, 3) + // We deleted the video (because unfederated), so even if we recreated it, the old entry is still deleted + await checkPlaylistElementType(group2, playlistServer1UUID2, VideoPlaylistElementType.DELETED, position, name, 3) + } + }) - const videos: Video[] = res.body.data - expect(videos).to.have.lengthOf(6) + it('Should update the element type if the account or server of the video is blocked', async function () { + this.timeout(90000) - expect(videos[0].name).to.equal('video 0 server 1') - expect(videos[0].playlistElement.position).to.equal(1) - expect(videos[0].playlistElement.startTimestamp).to.equal(15) - expect(videos[0].playlistElement.stopTimestamp).to.equal(28) + const name = 'video 90' + const position = 2 - expect(videos[1].name).to.equal('video 1 server 3') - expect(videos[1].playlistElement.position).to.equal(2) - expect(videos[1].playlistElement.startTimestamp).to.equal(35) - expect(videos[1].playlistElement.stopTimestamp).to.be.null + { + await addAccountToAccountBlocklist(servers[ 0 ].url, userAccessTokenServer1, 'root@localhost:' + servers[1].port) + await waitJobs(servers) - expect(videos[2].name).to.equal('video 2 server 3') - expect(videos[2].playlistElement.position).to.equal(3) - expect(videos[2].playlistElement.startTimestamp).to.be.null - expect(videos[2].playlistElement.stopTimestamp).to.be.null + await checkPlaylistElementType(groupUser1, playlistServer1UUID2, VideoPlaylistElementType.UNAVAILABLE, position, name, 3) + await checkPlaylistElementType(group2, playlistServer1UUID2, VideoPlaylistElementType.REGULAR, position, name, 3) - expect(videos[3].name).to.equal('video 3 server 1') - expect(videos[3].playlistElement.position).to.equal(4) - expect(videos[3].playlistElement.startTimestamp).to.be.null - expect(videos[3].playlistElement.stopTimestamp).to.equal(35) + await removeAccountFromAccountBlocklist(servers[ 0 ].url, userAccessTokenServer1, 'root@localhost:' + servers[1].port) + await waitJobs(servers) - expect(videos[4].name).to.equal('video 4 server 1') - expect(videos[4].playlistElement.position).to.equal(5) - expect(videos[4].playlistElement.startTimestamp).to.equal(45) - expect(videos[4].playlistElement.stopTimestamp).to.equal(60) + await checkPlaylistElementType(group2, playlistServer1UUID2, VideoPlaylistElementType.REGULAR, position, name, 3) + } - expect(videos[5].name).to.equal('NSFW video') - expect(videos[5].playlistElement.position).to.equal(6) - expect(videos[5].playlistElement.startTimestamp).to.equal(5) - expect(videos[5].playlistElement.stopTimestamp).to.be.null + { + await addServerToAccountBlocklist(servers[ 0 ].url, userAccessTokenServer1, 'localhost:' + servers[1].port) + await waitJobs(servers) - const res2 = await getPlaylistVideos(server.url, server.accessToken, playlistServer1UUID, 0, 10, { nsfw: false }) - expect(res2.body.total).to.equal(5) - expect(res2.body.data.find(v => v.name === 'NSFW video')).to.be.undefined + await checkPlaylistElementType(groupUser1, playlistServer1UUID2, VideoPlaylistElementType.UNAVAILABLE, position, name, 3) + await checkPlaylistElementType(group2, playlistServer1UUID2, VideoPlaylistElementType.REGULAR, position, name, 3) - const res3 = await getPlaylistVideos(server.url, server.accessToken, playlistServer1UUID, 0, 2) - expect(res3.body.data).to.have.lengthOf(2) - } - }) + await removeServerFromAccountBlocklist(servers[ 0 ].url, userAccessTokenServer1, 'localhost:' + servers[1].port) + await waitJobs(servers) - it('Should reorder the playlist', async function () { - this.timeout(30000) + await checkPlaylistElementType(group2, playlistServer1UUID2, VideoPlaylistElementType.REGULAR, position, name, 3) + } - { - await reorderVideosPlaylist({ - url: servers[ 0 ].url, - token: servers[ 0 ].accessToken, - playlistId: playlistServer1Id, - elementAttrs: { - startPosition: 2, - insertAfterPosition: 3 - } - }) + { + await addAccountToServerBlocklist(servers[ 0 ].url, servers[ 0 ].accessToken, 'root@localhost:' + servers[1].port) + await waitJobs(servers) - await waitJobs(servers) + await checkPlaylistElementType(groupUser1, playlistServer1UUID2, VideoPlaylistElementType.UNAVAILABLE, position, name, 3) + await checkPlaylistElementType(group2, playlistServer1UUID2, VideoPlaylistElementType.REGULAR, position, name, 3) - for (const server of servers) { - const res = await getPlaylistVideos(server.url, server.accessToken, playlistServer1UUID, 0, 10) - const names = res.body.data.map(v => v.name) + await removeAccountFromServerBlocklist(servers[ 0 ].url, servers[ 0 ].accessToken, 'root@localhost:' + servers[1].port) + await waitJobs(servers) - expect(names).to.deep.equal([ - 'video 0 server 1', - 'video 2 server 3', - 'video 1 server 3', - 'video 3 server 1', - 'video 4 server 1', - 'NSFW video' - ]) + await checkPlaylistElementType(group2, playlistServer1UUID2, VideoPlaylistElementType.REGULAR, position, name, 3) } - } - { - await reorderVideosPlaylist({ - url: servers[0].url, - token: servers[0].accessToken, - playlistId: playlistServer1Id, - elementAttrs: { - startPosition: 1, - reorderLength: 3, - insertAfterPosition: 4 - } - }) + { + await addServerToServerBlocklist(servers[ 0 ].url, servers[ 0 ].accessToken, 'localhost:' + servers[1].port) + await waitJobs(servers) - await waitJobs(servers) + await checkPlaylistElementType(groupUser1, playlistServer1UUID2, VideoPlaylistElementType.UNAVAILABLE, position, name, 3) + await checkPlaylistElementType(group2, playlistServer1UUID2, VideoPlaylistElementType.REGULAR, position, name, 3) - for (const server of servers) { - const res = await getPlaylistVideos(server.url, server.accessToken, playlistServer1UUID, 0, 10) - const names = res.body.data.map(v => v.name) + await removeServerFromServerBlocklist(servers[ 0 ].url, servers[ 0 ].accessToken, 'localhost:' + servers[1].port) + await waitJobs(servers) - expect(names).to.deep.equal([ - 'video 3 server 1', - 'video 0 server 1', - 'video 2 server 3', - 'video 1 server 3', - 'video 4 server 1', - 'NSFW video' - ]) + await checkPlaylistElementType(group2, playlistServer1UUID2, VideoPlaylistElementType.REGULAR, position, name, 3) } - } - - { - await reorderVideosPlaylist({ - url: servers[0].url, - token: servers[0].accessToken, - playlistId: playlistServer1Id, - elementAttrs: { - startPosition: 6, - insertAfterPosition: 3 - } - }) + }) - await waitJobs(servers) + it('Should hide the video if it is NSFW', async function () { + const res = await getPlaylistVideos(servers[0].url, userAccessTokenServer1, playlistServer1UUID2, 0, 10, { nsfw: false }) + expect(res.body.total).to.equal(3) - for (const server of servers) { - const res = await getPlaylistVideos(server.url, server.accessToken, playlistServer1UUID, 0, 10) - const videos: Video[] = res.body.data + const elements: VideoPlaylistElement[] = res.body.data + const element = elements.find(e => e.position === 3) - const names = videos.map(v => v.name) + expect(element).to.exist + expect(element.video).to.be.null + expect(element.type).to.equal(VideoPlaylistElementType.UNAVAILABLE) + }) - expect(names).to.deep.equal([ - 'video 3 server 1', - 'video 0 server 1', - 'video 2 server 3', - 'NSFW video', - 'video 1 server 3', - 'video 4 server 1' - ]) + }) - for (let i = 1; i <= videos.length; i++) { - expect(videos[i - 1].playlistElement.position).to.equal(i) + describe('Managing playlist elements', function () { + + it('Should reorder the playlist', async function () { + this.timeout(30000) + + { + await reorderVideosPlaylist({ + url: servers[ 0 ].url, + token: servers[ 0 ].accessToken, + playlistId: playlistServer1Id, + elementAttrs: { + startPosition: 2, + insertAfterPosition: 3 + } + }) + + await waitJobs(servers) + + for (const server of servers) { + const res = await getPlaylistVideos(server.url, server.accessToken, playlistServer1UUID, 0, 10) + const names = (res.body.data as VideoPlaylistElement[]).map(v => v.video.name) + + expect(names).to.deep.equal([ + 'video 0 server 1', + 'video 2 server 3', + 'video 1 server 3', + 'video 3 server 1', + 'video 4 server 1', + 'NSFW video' + ]) } } - } - }) - - it('Should update startTimestamp/endTimestamp of some elements', async function () { - this.timeout(30000) - await updateVideoPlaylistElement({ - url: servers[0].url, - token: servers[0].accessToken, - playlistId: playlistServer1Id, - videoId: servers[0].videos[3].uuid, - elementAttrs: { - startTimestamp: 1 + { + await reorderVideosPlaylist({ + url: servers[ 0 ].url, + token: servers[ 0 ].accessToken, + playlistId: playlistServer1Id, + elementAttrs: { + startPosition: 1, + reorderLength: 3, + insertAfterPosition: 4 + } + }) + + await waitJobs(servers) + + for (const server of servers) { + const res = await getPlaylistVideos(server.url, server.accessToken, playlistServer1UUID, 0, 10) + const names = (res.body.data as VideoPlaylistElement[]).map(v => v.video.name) + + expect(names).to.deep.equal([ + 'video 3 server 1', + 'video 0 server 1', + 'video 2 server 3', + 'video 1 server 3', + 'video 4 server 1', + 'NSFW video' + ]) + } } - }) - await updateVideoPlaylistElement({ - url: servers[0].url, - token: servers[0].accessToken, - playlistId: playlistServer1Id, - videoId: servers[0].videos[4].uuid, - elementAttrs: { - stopTimestamp: null + { + await reorderVideosPlaylist({ + url: servers[ 0 ].url, + token: servers[ 0 ].accessToken, + playlistId: playlistServer1Id, + elementAttrs: { + startPosition: 6, + insertAfterPosition: 3 + } + }) + + await waitJobs(servers) + + for (const server of servers) { + const res = await getPlaylistVideos(server.url, server.accessToken, playlistServer1UUID, 0, 10) + const elements: VideoPlaylistElement[] = res.body.data + const names = elements.map(v => v.video.name) + + expect(names).to.deep.equal([ + 'video 3 server 1', + 'video 0 server 1', + 'video 2 server 3', + 'NSFW video', + 'video 1 server 3', + 'video 4 server 1' + ]) + + for (let i = 1; i <= elements.length; i++) { + expect(elements[ i - 1 ].position).to.equal(i) + } + } } }) - await waitJobs(servers) + it('Should update startTimestamp/endTimestamp of some elements', async function () { + this.timeout(30000) + + await updateVideoPlaylistElement({ + url: servers[ 0 ].url, + token: servers[ 0 ].accessToken, + playlistId: playlistServer1Id, + playlistElementId: playlistElementServer1Video4, + elementAttrs: { + startTimestamp: 1 + } + }) - for (const server of servers) { - const res = await getPlaylistVideos(server.url, server.accessToken, playlistServer1UUID, 0, 10) - const videos: Video[] = res.body.data + await updateVideoPlaylistElement({ + url: servers[ 0 ].url, + token: servers[ 0 ].accessToken, + playlistId: playlistServer1Id, + playlistElementId: playlistElementServer1Video5, + elementAttrs: { + stopTimestamp: null + } + }) - expect(videos[0].name).to.equal('video 3 server 1') - expect(videos[0].playlistElement.position).to.equal(1) - expect(videos[0].playlistElement.startTimestamp).to.equal(1) - expect(videos[0].playlistElement.stopTimestamp).to.equal(35) + await waitJobs(servers) - expect(videos[5].name).to.equal('video 4 server 1') - expect(videos[5].playlistElement.position).to.equal(6) - expect(videos[5].playlistElement.startTimestamp).to.equal(45) - expect(videos[5].playlistElement.stopTimestamp).to.be.null - } - }) + for (const server of servers) { + const res = await getPlaylistVideos(server.url, server.accessToken, playlistServer1UUID, 0, 10) + const elements: VideoPlaylistElement[] = res.body.data - it('Should check videos existence in my playlist', async function () { - const videoIds = [ - servers[0].videos[0].id, - 42000, - servers[0].videos[3].id, - 43000, - servers[0].videos[4].id - ] - const res = await doVideosExistInMyPlaylist(servers[ 0 ].url, servers[ 0 ].accessToken, videoIds) - const obj = res.body as VideoExistInPlaylist + expect(elements[ 0 ].video.name).to.equal('video 3 server 1') + expect(elements[ 0 ].position).to.equal(1) + expect(elements[ 0 ].startTimestamp).to.equal(1) + expect(elements[ 0 ].stopTimestamp).to.equal(35) - { - const elem = obj[servers[0].videos[0].id] - expect(elem).to.have.lengthOf(1) - expect(elem[ 0 ].playlistId).to.equal(playlistServer1Id) - expect(elem[ 0 ].startTimestamp).to.equal(15) - expect(elem[ 0 ].stopTimestamp).to.equal(28) - } + expect(elements[ 5 ].video.name).to.equal('video 4 server 1') + expect(elements[ 5 ].position).to.equal(6) + expect(elements[ 5 ].startTimestamp).to.equal(45) + expect(elements[ 5 ].stopTimestamp).to.be.null + } + }) - { - const elem = obj[servers[0].videos[3].id] - expect(elem).to.have.lengthOf(1) - expect(elem[ 0 ].playlistId).to.equal(playlistServer1Id) - expect(elem[ 0 ].startTimestamp).to.equal(1) - expect(elem[ 0 ].stopTimestamp).to.equal(35) - } + it('Should check videos existence in my playlist', async function () { + const videoIds = [ + servers[ 0 ].videos[ 0 ].id, + 42000, + servers[ 0 ].videos[ 3 ].id, + 43000, + servers[ 0 ].videos[ 4 ].id + ] + const res = await doVideosExistInMyPlaylist(servers[ 0 ].url, servers[ 0 ].accessToken, videoIds) + const obj = res.body as VideoExistInPlaylist + + { + const elem = obj[ servers[ 0 ].videos[ 0 ].id ] + expect(elem).to.have.lengthOf(1) + expect(elem[ 0 ].playlistElementId).to.exist + expect(elem[ 0 ].playlistId).to.equal(playlistServer1Id) + expect(elem[ 0 ].startTimestamp).to.equal(15) + expect(elem[ 0 ].stopTimestamp).to.equal(28) + } - { - const elem = obj[servers[0].videos[4].id] - expect(elem).to.have.lengthOf(1) - expect(elem[ 0 ].playlistId).to.equal(playlistServer1Id) - expect(elem[ 0 ].startTimestamp).to.equal(45) - expect(elem[ 0 ].stopTimestamp).to.equal(null) - } + { + const elem = obj[ servers[ 0 ].videos[ 3 ].id ] + expect(elem).to.have.lengthOf(1) + expect(elem[ 0 ].playlistElementId).to.equal(playlistElementServer1Video4) + expect(elem[ 0 ].playlistId).to.equal(playlistServer1Id) + expect(elem[ 0 ].startTimestamp).to.equal(1) + expect(elem[ 0 ].stopTimestamp).to.equal(35) + } - expect(obj[42000]).to.have.lengthOf(0) - expect(obj[43000]).to.have.lengthOf(0) - }) + { + const elem = obj[ servers[ 0 ].videos[ 4 ].id ] + expect(elem).to.have.lengthOf(1) + expect(elem[ 0 ].playlistId).to.equal(playlistServer1Id) + expect(elem[ 0 ].startTimestamp).to.equal(45) + expect(elem[ 0 ].stopTimestamp).to.equal(null) + } - it('Should automatically update updatedAt field of playlists', async function () { - const server = servers[1] - const videoId = servers[1].videos[5].id + expect(obj[ 42000 ]).to.have.lengthOf(0) + expect(obj[ 43000 ]).to.have.lengthOf(0) + }) - async function getPlaylistNames () { - const res = await getAccountPlaylistsListWithToken(server.url, server.accessToken, 'root', 0, 5, undefined, '-updatedAt') + it('Should automatically update updatedAt field of playlists', async function () { + const server = servers[ 1 ] + const videoId = servers[ 1 ].videos[ 5 ].id - return (res.body.data as VideoPlaylist[]).map(p => p.displayName) - } + async function getPlaylistNames () { + const res = await getAccountPlaylistsListWithToken(server.url, server.accessToken, 'root', 0, 5, undefined, '-updatedAt') - const elementAttrs = { videoId } - await addVideoInPlaylist({ url: server.url, token: server.accessToken, playlistId: playlistServer2Id1, elementAttrs }) - await addVideoInPlaylist({ url: server.url, token: server.accessToken, playlistId: playlistServer2Id2, elementAttrs }) + return (res.body.data as VideoPlaylist[]).map(p => p.displayName) + } - const names1 = await getPlaylistNames() - expect(names1[0]).to.equal('playlist 3 updated') - expect(names1[1]).to.equal('playlist 2') + const elementAttrs = { videoId } + const res1 = await addVideoInPlaylist({ url: server.url, token: server.accessToken, playlistId: playlistServer2Id1, elementAttrs }) + const res2 = await addVideoInPlaylist({ url: server.url, token: server.accessToken, playlistId: playlistServer2Id2, elementAttrs }) - await removeVideoFromPlaylist({ url: server.url, token: server.accessToken, playlistId: playlistServer2Id1, videoId }) + const element1 = res1.body.videoPlaylistElement.id + const element2 = res2.body.videoPlaylistElement.id - const names2 = await getPlaylistNames() - expect(names2[0]).to.equal('playlist 2') - expect(names2[1]).to.equal('playlist 3 updated') + const names1 = await getPlaylistNames() + expect(names1[ 0 ]).to.equal('playlist 3 updated') + expect(names1[ 1 ]).to.equal('playlist 2') - await removeVideoFromPlaylist({ url: server.url, token: server.accessToken, playlistId: playlistServer2Id2, videoId }) + await removeVideoFromPlaylist({ + url: server.url, + token: server.accessToken, + playlistId: playlistServer2Id1, + playlistElementId: element1 + }) - const names3 = await getPlaylistNames() - expect(names3[0]).to.equal('playlist 3 updated') - expect(names3[1]).to.equal('playlist 2') - }) + const names2 = await getPlaylistNames() + expect(names2[ 0 ]).to.equal('playlist 2') + expect(names2[ 1 ]).to.equal('playlist 3 updated') - it('Should delete some elements', async function () { - this.timeout(30000) + await removeVideoFromPlaylist({ + url: server.url, + token: server.accessToken, + playlistId: playlistServer2Id2, + playlistElementId: element2 + }) - await removeVideoFromPlaylist({ - url: servers[0].url, - token: servers[0].accessToken, - playlistId: playlistServer1Id, - videoId: servers[0].videos[3].uuid + const names3 = await getPlaylistNames() + expect(names3[ 0 ]).to.equal('playlist 3 updated') + expect(names3[ 1 ]).to.equal('playlist 2') }) - await removeVideoFromPlaylist({ - url: servers[0].url, - token: servers[0].accessToken, - playlistId: playlistServer1Id, - videoId: nsfwVideoServer1 - }) + it('Should delete some elements', async function () { + this.timeout(30000) - await waitJobs(servers) + await removeVideoFromPlaylist({ + url: servers[ 0 ].url, + token: servers[ 0 ].accessToken, + playlistId: playlistServer1Id, + playlistElementId: playlistElementServer1Video4 + }) - for (const server of servers) { - const res = await getPlaylistVideos(server.url, server.accessToken, playlistServer1UUID, 0, 10) + await removeVideoFromPlaylist({ + url: servers[ 0 ].url, + token: servers[ 0 ].accessToken, + playlistId: playlistServer1Id, + playlistElementId: playlistElementNSFW + }) - expect(res.body.total).to.equal(4) + await waitJobs(servers) - const videos: Video[] = res.body.data - expect(videos).to.have.lengthOf(4) + for (const server of servers) { + const res = await getPlaylistVideos(server.url, server.accessToken, playlistServer1UUID, 0, 10) - expect(videos[ 0 ].name).to.equal('video 0 server 1') - expect(videos[ 0 ].playlistElement.position).to.equal(1) + expect(res.body.total).to.equal(4) - expect(videos[ 1 ].name).to.equal('video 2 server 3') - expect(videos[ 1 ].playlistElement.position).to.equal(2) + const elements: VideoPlaylistElement[] = res.body.data + expect(elements).to.have.lengthOf(4) - expect(videos[ 2 ].name).to.equal('video 1 server 3') - expect(videos[ 2 ].playlistElement.position).to.equal(3) + expect(elements[ 0 ].video.name).to.equal('video 0 server 1') + expect(elements[ 0 ].position).to.equal(1) - expect(videos[ 3 ].name).to.equal('video 4 server 1') - expect(videos[ 3 ].playlistElement.position).to.equal(4) - } - }) + expect(elements[ 1 ].video.name).to.equal('video 2 server 3') + expect(elements[ 1 ].position).to.equal(2) - it('Should be able to create a public playlist, and set it to private', async function () { - this.timeout(30000) + expect(elements[ 2 ].video.name).to.equal('video 1 server 3') + expect(elements[ 2 ].position).to.equal(3) - const res = await createVideoPlaylist({ - url: servers[0].url, - token: servers[0].accessToken, - playlistAttrs: { - displayName: 'my super public playlist', - privacy: VideoPlaylistPrivacy.PUBLIC, - videoChannelId: servers[0].videoChannel.id + expect(elements[ 3 ].video.name).to.equal('video 4 server 1') + expect(elements[ 3 ].position).to.equal(4) } }) - const videoPlaylistIds = res.body.videoPlaylist - await waitJobs(servers) + it('Should be able to create a public playlist, and set it to private', async function () { + this.timeout(30000) - for (const server of servers) { - await getVideoPlaylist(server.url, videoPlaylistIds.uuid, 200) - } + const res = await createVideoPlaylist({ + url: servers[ 0 ].url, + token: servers[ 0 ].accessToken, + playlistAttrs: { + displayName: 'my super public playlist', + privacy: VideoPlaylistPrivacy.PUBLIC, + videoChannelId: servers[ 0 ].videoChannel.id + } + }) + const videoPlaylistIds = res.body.videoPlaylist - const playlistAttrs = { privacy: VideoPlaylistPrivacy.PRIVATE } - await updateVideoPlaylist({ url: servers[0].url, token: servers[0].accessToken, playlistId: videoPlaylistIds.id, playlistAttrs }) + await waitJobs(servers) - await waitJobs(servers) + for (const server of servers) { + await getVideoPlaylist(server.url, videoPlaylistIds.uuid, 200) + } - for (const server of [ servers[1], servers[2] ]) { - await getVideoPlaylist(server.url, videoPlaylistIds.uuid, 404) - } - await getVideoPlaylist(servers[0].url, videoPlaylistIds.uuid, 401) + const playlistAttrs = { privacy: VideoPlaylistPrivacy.PRIVATE } + await updateVideoPlaylist({ url: servers[ 0 ].url, token: servers[ 0 ].accessToken, playlistId: videoPlaylistIds.id, playlistAttrs }) - await getVideoPlaylistWithToken(servers[0].url, servers[0].accessToken, videoPlaylistIds.uuid, 200) - }) + await waitJobs(servers) - it('Should delete the playlist on server 1 and delete on server 2 and 3', async function () { - this.timeout(30000) + for (const server of [ servers[ 1 ], servers[ 2 ] ]) { + await getVideoPlaylist(server.url, videoPlaylistIds.uuid, 404) + } + await getVideoPlaylist(servers[ 0 ].url, videoPlaylistIds.uuid, 401) - await deleteVideoPlaylist(servers[0].url, servers[0].accessToken, playlistServer1Id) + await getVideoPlaylistWithToken(servers[ 0 ].url, servers[ 0 ].accessToken, videoPlaylistIds.uuid, 200) + }) + }) - await waitJobs(servers) + describe('Playlist deletion', function () { - for (const server of servers) { - await getVideoPlaylist(server.url, playlistServer1UUID, 404) - } - }) + it('Should delete the playlist on server 1 and delete on server 2 and 3', async function () { + this.timeout(30000) - it('Should have deleted the thumbnail on server 1, 2 and 3', async function () { - this.timeout(30000) + await deleteVideoPlaylist(servers[ 0 ].url, servers[ 0 ].accessToken, playlistServer1Id) - for (const server of servers) { - await checkPlaylistFilesWereRemoved(playlistServer1UUID, server.internalServerNumber) - } - }) + await waitJobs(servers) - it('Should unfollow servers 1 and 2 and hide their playlists', async function () { - this.timeout(30000) + for (const server of servers) { + await getVideoPlaylist(server.url, playlistServer1UUID, 404) + } + }) - const finder = data => data.find(p => p.displayName === 'my super playlist') + it('Should have deleted the thumbnail on server 1, 2 and 3', async function () { + this.timeout(30000) - { - const res = await getVideoPlaylistsList(servers[ 2 ].url, 0, 5) - expect(res.body.total).to.equal(2) - expect(finder(res.body.data)).to.not.be.undefined - } + for (const server of servers) { + await checkPlaylistFilesWereRemoved(playlistServer1UUID, server.internalServerNumber) + } + }) - await unfollow(servers[2].url, servers[2].accessToken, servers[0]) + it('Should unfollow servers 1 and 2 and hide their playlists', async function () { + this.timeout(30000) - { - const res = await getVideoPlaylistsList(servers[ 2 ].url, 0, 5) - expect(res.body.total).to.equal(1) + const finder = data => data.find(p => p.displayName === 'my super playlist') - expect(finder(res.body.data)).to.be.undefined - } - }) + { + const res = await getVideoPlaylistsList(servers[ 2 ].url, 0, 5) + expect(res.body.total).to.equal(3) + expect(finder(res.body.data)).to.not.be.undefined + } - it('Should delete a channel and put the associated playlist in private mode', async function () { - this.timeout(30000) + await unfollow(servers[ 2 ].url, servers[ 2 ].accessToken, servers[ 0 ]) - const res = await addVideoChannel(servers[0].url, servers[0].accessToken, { name: 'super_channel', displayName: 'super channel' }) - const videoChannelId = res.body.videoChannel.id + { + const res = await getVideoPlaylistsList(servers[ 2 ].url, 0, 5) + expect(res.body.total).to.equal(1) - const res2 = await createVideoPlaylist({ - url: servers[0].url, - token: servers[0].accessToken, - playlistAttrs: { - displayName: 'channel playlist', - privacy: VideoPlaylistPrivacy.PUBLIC, - videoChannelId + expect(finder(res.body.data)).to.be.undefined } }) - const videoPlaylistUUID = res2.body.videoPlaylist.uuid - await waitJobs(servers) + it('Should delete a channel and put the associated playlist in private mode', async function () { + this.timeout(30000) - await deleteVideoChannel(servers[0].url, servers[0].accessToken, 'super_channel') + const res = await addVideoChannel(servers[ 0 ].url, servers[ 0 ].accessToken, { name: 'super_channel', displayName: 'super channel' }) + const videoChannelId = res.body.videoChannel.id - await waitJobs(servers) + const res2 = await createVideoPlaylist({ + url: servers[ 0 ].url, + token: servers[ 0 ].accessToken, + playlistAttrs: { + displayName: 'channel playlist', + privacy: VideoPlaylistPrivacy.PUBLIC, + videoChannelId + } + }) + const videoPlaylistUUID = res2.body.videoPlaylist.uuid + + await waitJobs(servers) - const res3 = await getVideoPlaylistWithToken(servers[0].url, servers[0].accessToken, videoPlaylistUUID) - expect(res3.body.displayName).to.equal('channel playlist') - expect(res3.body.privacy.id).to.equal(VideoPlaylistPrivacy.PRIVATE) + await deleteVideoChannel(servers[ 0 ].url, servers[ 0 ].accessToken, 'super_channel') - await getVideoPlaylist(servers[1].url, videoPlaylistUUID, 404) - }) + await waitJobs(servers) - it('Should delete an account and delete its playlists', async function () { - this.timeout(30000) + const res3 = await getVideoPlaylistWithToken(servers[ 0 ].url, servers[ 0 ].accessToken, videoPlaylistUUID) + expect(res3.body.displayName).to.equal('channel playlist') + expect(res3.body.privacy.id).to.equal(VideoPlaylistPrivacy.PRIVATE) - const user = { username: 'user_1', password: 'password' } - const res = await createUser({ - url: servers[ 0 ].url, - accessToken: servers[ 0 ].accessToken, - username: user.username, - password: user.password + await getVideoPlaylist(servers[ 1 ].url, videoPlaylistUUID, 404) }) - const userId = res.body.user.id - const userAccessToken = await userLogin(servers[0], user) + it('Should delete an account and delete its playlists', async function () { + this.timeout(30000) - const resChannel = await getMyUserInformation(servers[0].url, userAccessToken) - const userChannel = (resChannel.body as User).videoChannels[0] + const user = { username: 'user_1', password: 'password' } + const res = await createUser({ + url: servers[ 0 ].url, + accessToken: servers[ 0 ].accessToken, + username: user.username, + password: user.password + }) - await createVideoPlaylist({ - url: servers[0].url, - token: userAccessToken, - playlistAttrs: { - displayName: 'playlist to be deleted', - privacy: VideoPlaylistPrivacy.PUBLIC, - videoChannelId: userChannel.id - } - }) + const userId = res.body.user.id + const userAccessToken = await userLogin(servers[ 0 ], user) - await waitJobs(servers) + const resChannel = await getMyUserInformation(servers[ 0 ].url, userAccessToken) + const userChannel = (resChannel.body as User).videoChannels[ 0 ] - const finder = data => data.find(p => p.displayName === 'playlist to be deleted') + await createVideoPlaylist({ + url: servers[ 0 ].url, + token: userAccessToken, + playlistAttrs: { + displayName: 'playlist to be deleted', + privacy: VideoPlaylistPrivacy.PUBLIC, + videoChannelId: userChannel.id + } + }) - { - for (const server of [ servers[0], servers[1] ]) { - const res = await getVideoPlaylistsList(server.url, 0, 15) - expect(finder(res.body.data)).to.not.be.undefined + await waitJobs(servers) + + const finder = data => data.find(p => p.displayName === 'playlist to be deleted') + + { + for (const server of [ servers[ 0 ], servers[ 1 ] ]) { + const res = await getVideoPlaylistsList(server.url, 0, 15) + expect(finder(res.body.data)).to.not.be.undefined + } } - } - await removeUser(servers[0].url, userId, servers[0].accessToken) - await waitJobs(servers) + await removeUser(servers[ 0 ].url, userId, servers[ 0 ].accessToken) + await waitJobs(servers) - { - for (const server of [ servers[0], servers[1] ]) { - const res = await getVideoPlaylistsList(server.url, 0, 15) - expect(finder(res.body.data)).to.be.undefined + { + for (const server of [ servers[ 0 ], servers[ 1 ] ]) { + const res = await getVideoPlaylistsList(server.url, 0, 15) + expect(finder(res.body.data)).to.be.undefined + } } - } + }) }) after(async function () { -- cgit v1.2.3