-/* tslint:disable:no-unused-expression */
+/* eslint-disable @typescript-eslint/no-unused-expressions,@typescript-eslint/require-await */
-import * as chai from 'chai'
-import 'mocha'
+import { expect } from 'chai'
+import { checkPlaylistFilesWereRemoved, testImage } from '@server/tests/shared'
+import { wait } from '@shared/core-utils'
+import {
+ HttpStatusCode,
+ VideoPlaylist,
+ VideoPlaylistCreateResult,
+ VideoPlaylistElementType,
+ VideoPlaylistPrivacy,
+ VideoPlaylistType,
+ VideoPrivacy
+} from '@shared/models'
import {
- addVideoChannel,
- addVideoInPlaylist,
- checkPlaylistFilesWereRemoved,
cleanupTests,
- createUser,
- createVideoPlaylist,
- deleteVideoChannel,
- deleteVideoPlaylist,
+ createMultipleServers,
doubleFollow,
- doVideosExistInMyPlaylist,
- flushAndRunMultipleServers,
- getAccountPlaylistsList,
- getAccountPlaylistsListWithToken,
- getMyUserInformation,
- getPlaylistVideos,
- getVideoChannelPlaylistsList,
- getVideoPlaylist,
- getVideoPlaylistPrivacies,
- getVideoPlaylistsList,
- getVideoPlaylistWithToken,
- removeUser,
- removeVideoFromPlaylist,
- reorderVideosPlaylist,
- ServerInfo,
+ PeerTubeServer,
+ PlaylistsCommand,
setAccessTokensToServers,
+ setDefaultAccountAvatar,
setDefaultVideoChannel,
- testImage,
- unfollow,
- updateVideoPlaylist,
- updateVideoPlaylistElement,
- uploadVideo,
- uploadVideoAndGetId,
- userLogin,
- waitJobs,
- generateUserAccessToken
-} 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 { 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'
-
-const expect = chai.expect
+ waitJobs
+} from '@shared/server-commands'
+import { uuidToShort } from '@shared/extra-utils'
+
+async function checkPlaylistElementType (
+ servers: PeerTubeServer[],
+ playlistId: string,
+ type: VideoPlaylistElementType,
+ position: number,
+ name: string,
+ total: number
+) {
+ for (const server of servers) {
+ const body = await server.playlists.listVideos({ token: server.accessToken, playlistId, start: 0, count: 10 })
+ expect(body.total).to.equal(total)
+
+ const videoElement = body.data.find(e => 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[] = []
+ let servers: PeerTubeServer[] = []
let playlistServer2Id1: number
let playlistServer2Id2: number
- let playlistServer2UUID2: number
+ let playlistServer2UUID2: string
let playlistServer1Id: number
+ let playlistServer1DisplayName: string
let playlistServer1UUID: string
+ let playlistServer1UUID2: string
+
+ let playlistElementServer1Video4: number
+ let playlistElementServer1Video5: number
+ let playlistElementNSFW: number
let nsfwVideoServer1: number
+ let userTokenServer1: string
+
+ let commands: PlaylistsCommand[]
+
before(async function () {
- this.timeout(120000)
+ this.timeout(240000)
- servers = await flushAndRunMultipleServers(3, { transcoding: { enabled: false } })
+ servers = await createMultipleServers(3)
// Get the access tokens
await setAccessTokensToServers(servers)
await setDefaultVideoChannel(servers)
+ await setDefaultAccountAvatar(servers)
+
+ for (const server of servers) {
+ await server.config.disableTranscoding()
+ }
// Server 1 and server 2 follow each other
await doubleFollow(servers[0], servers[1])
// Server 1 and server 3 follow each other
await doubleFollow(servers[0], servers[2])
+ commands = servers.map(s => s.playlists)
+
{
- const serverPromises: Promise<any>[][] = []
+ servers[0].store.videos = []
+ servers[1].store.videos = []
+ servers[2].store.videos = []
for (const server of servers) {
- const videoPromises: Promise<any>[] = []
-
for (let i = 0; i < 7; i++) {
- videoPromises.push(
- uploadVideo(server.url, server.accessToken, { name: `video ${i} server ${server.serverNumber}`, nsfw: false })
- .then(res => res.body.video)
- )
- }
+ const name = `video ${i} server ${server.serverNumber}`
+ const video = await server.videos.upload({ attributes: { name, nsfw: false } })
- serverPromises.push(videoPromises)
+ server.store.videos.push(video)
+ }
}
-
- servers[0].videos = await Promise.all(serverPromises[0])
- servers[1].videos = await Promise.all(serverPromises[1])
- servers[2].videos = await Promise.all(serverPromises[2])
}
- nsfwVideoServer1 = (await uploadVideoAndGetId({ server: servers[ 0 ], videoName: 'NSFW video', nsfw: true })).id
+ nsfwVideoServer1 = (await servers[0].videos.quickUpload({ name: 'NSFW video', nsfw: true })).id
+
+ userTokenServer1 = await servers[0].users.generateUserAndToken('user1')
await waitJobs(servers)
})
- it('Should list video playlist privacies', async function () {
- const res = await getVideoPlaylistPrivacies(servers[0].url)
+ describe('Get default playlists', function () {
- const privacies = res.body
- expect(Object.keys(privacies)).to.have.length.at.least(3)
+ it('Should list video playlist privacies', async function () {
+ const privacies = await commands[0].getPrivacies()
- expect(privacies[3]).to.equal('Private')
- })
+ expect(Object.keys(privacies)).to.have.length.at.least(3)
+ 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 token = servers[0].accessToken
- {
- const res = await getAccountPlaylistsListWithToken(url, accessToken, 'root', 0, 5, VideoPlaylistType.WATCH_LATER)
+ {
+ const body = await commands[0].listByAccount({ token, handle: 'root', playlistType: VideoPlaylistType.WATCH_LATER })
- expect(res.body.total).to.equal(1)
- expect(res.body.data).to.have.lengthOf(1)
+ expect(body.total).to.equal(1)
+ expect(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 playlist = 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)
+ {
+ const body = await commands[0].listByAccount({ token, handle: 'root', playlistType: VideoPlaylistType.REGULAR })
- expect(res.body.total).to.equal(0)
- expect(res.body.data).to.have.lengthOf(0)
- }
+ expect(body.total).to.equal(0)
+ expect(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)
- }
- })
+ {
+ const body = await commands[0].listByAccount({ handle: 'root' })
+ expect(body.total).to.equal(0)
+ expect(body.data).to.have.lengthOf(0)
+ }
+ })
- it('Should get private playlist for a classic user', async function () {
- const token = await generateUserAccessToken(servers[0], 'toto')
+ it('Should get private playlist for a classic user', async function () {
+ const token = await servers[0].users.generateUserAndToken('toto')
- const res = await getAccountPlaylistsListWithToken(servers[0].url, token, 'toto', 0, 5)
+ const body = await commands[0].listByAccount({ token, handle: 'toto' })
- expect(res.body.total).to.equal(1)
- expect(res.body.data).to.have.lengthOf(1)
+ expect(body.total).to.equal(1)
+ expect(body.data).to.have.lengthOf(1)
- const playlistId = res.body.data[0].id
- await getPlaylistVideos(servers[0].url, token, playlistId, 0, 5)
+ const playlistId = body.data[0].id
+ await commands[0].listVideos({ token, playlistId })
+ })
})
- it('Should create a playlist on server 1 and have the playlist on server 2 and 3', async function () {
- this.timeout(30000)
+ describe('Create and federate playlists', function () {
- 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
- }
- })
+ it('Should create a playlist on server 1 and have the playlist on server 2 and 3', async function () {
+ this.timeout(30000)
- await waitJobs(servers)
+ await commands[0].create({
+ attributes: {
+ displayName: 'my super playlist',
+ privacy: VideoPlaylistPrivacy.PUBLIC,
+ description: 'my super description',
+ thumbnailfile: 'thumbnail.jpg',
+ videoChannelId: servers[0].store.channel.id
+ }
+ })
- 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)
+ await waitJobs(servers)
+ // Processing a playlist by the receiver could be long
+ await wait(3000)
- const playlistFromList = res.body.data[0] as VideoPlaylist
+ for (const server of servers) {
+ const body = await server.playlists.list({ start: 0, count: 5 })
+ expect(body.total).to.equal(1)
+ expect(body.data).to.have.lengthOf(1)
- const res2 = await getVideoPlaylist(server.url, playlistFromList.uuid)
- const playlistFromGet = res2.body
+ const playlistFromList = body.data[0]
- for (const playlist of [ playlistFromGet, playlistFromList ]) {
- expect(playlist.id).to.be.a('number')
- expect(playlist.uuid).to.be.a('string')
+ const playlistFromGet = await server.playlists.get({ playlistId: playlistFromList.uuid })
- expect(playlist.isLocal).to.equal(server.serverNumber === 1)
+ for (const playlist of [ playlistFromGet, playlistFromList ]) {
+ expect(playlist.id).to.be.a('number')
+ expect(playlist.uuid).to.be.a('string')
- 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.isLocal).to.equal(server.serverNumber === 1)
- expect(playlist.videosLength).to.equal(0)
+ 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.embedPath).to.equal('/video-playlists/embed/' + playlist.uuid)
- 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 create a playlist on server 2 and have the playlist on server 1 but not on server 3', async function () {
- this.timeout(30000)
+ expect(playlist.videosLength).to.equal(0)
- {
- const res = await createVideoPlaylist({
- url: servers[1].url,
- token: servers[1].accessToken,
- playlistAttrs: {
- displayName: 'playlist 2',
- privacy: VideoPlaylistPrivacy.PUBLIC,
- videoChannelId: servers[1].videoChannel.id
+ 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')
}
- })
- playlistServer2Id1 = res.body.videoPlaylist.id
- }
+ }
+ })
- {
- 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
- }
- })
+ 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 playlist = await servers[1].playlists.create({
+ attributes: {
+ displayName: 'playlist 2',
+ privacy: VideoPlaylistPrivacy.PUBLIC,
+ videoChannelId: servers[1].store.channel.id
+ }
+ })
+ playlistServer2Id1 = playlist.id
+ }
- playlistServer2Id2 = res.body.videoPlaylist.id
- playlistServer2UUID2 = res.body.videoPlaylist.uuid
- }
+ {
+ const playlist = await servers[1].playlists.create({
+ attributes: {
+ displayName: 'playlist 3',
+ privacy: VideoPlaylistPrivacy.PUBLIC,
+ thumbnailfile: 'thumbnail.jpg',
+ videoChannelId: servers[1].store.channel.id
+ }
+ })
+
+ playlistServer2Id2 = playlist.id
+ playlistServer2UUID2 = playlist.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 }
- })
- }
+ for (const id of [ playlistServer2Id1, playlistServer2Id2 ]) {
+ await servers[1].playlists.addElement({
+ playlistId: id,
+ attributes: { videoId: servers[1].store.videos[0].id, startTimestamp: 1, stopTimestamp: 2 }
+ })
+ await servers[1].playlists.addElement({
+ playlistId: id,
+ attributes: { videoId: servers[1].store.videos[1].id }
+ })
+ }
- await waitJobs(servers)
+ await waitJobs(servers)
+ await wait(3000)
- for (const server of [ servers[0], servers[1] ]) {
- const res = await getVideoPlaylistsList(server.url, 0, 5)
+ for (const server of [ servers[0], servers[1] ]) {
+ const body = await server.playlists.list({ start: 0, count: 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 playlist2 = 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 playlist3 = body.data.find(p => p.displayName === 'playlist 3')
+ expect(playlist3).to.not.be.undefined
+ await testImage(server.url, 'thumbnail', playlist3.thumbnailPath)
+ }
- 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
- })
+ const body = await servers[2].playlists.list({ start: 0, count: 5 })
+ expect(body.data.find(p => p.displayName === 'playlist 2')).to.be.undefined
+ expect(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)
+ 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])
+ // Server 2 and server 3 follow each other
+ await doubleFollow(servers[1], servers[2])
- const res = await getVideoPlaylistsList(servers[2].url, 0, 5)
+ const body = await servers[2].playlists.list({ start: 0, count: 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)
+ const playlist2 = 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
+ expect(body.data.find(p => p.displayName === 'playlist 3')).to.not.be.undefined
+ })
})
- it('Should correctly list the playlists', async function () {
- this.timeout(30000)
+ describe('List playlists', function () {
- {
- const res = await getVideoPlaylistsList(servers[ 2 ].url, 1, 2, 'createdAt')
+ it('Should correctly list the playlists', async function () {
+ this.timeout(30000)
- expect(res.body.total).to.equal(3)
+ {
+ const body = await servers[2].playlists.list({ start: 1, count: 2, sort: 'createdAt' })
+ expect(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 data = 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')
+ {
+ const body = await servers[2].playlists.list({ start: 1, count: 2, sort: '-createdAt' })
+ expect(body.total).to.equal(3)
- expect(res.body.total).to.equal(3)
+ const data = 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 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')
- }
- })
+ it('Should list video channel playlists', async function () {
+ this.timeout(30000)
- it('Should list video channel playlists', async function () {
- this.timeout(30000)
+ {
+ const body = await commands[0].listByChannel({ handle: 'root_channel', start: 0, count: 2, sort: '-createdAt' })
+ expect(body.total).to.equal(1)
- {
- const res = await getVideoChannelPlaylistsList(servers[ 0 ].url, 'root_channel', 0, 2, '-createdAt')
+ const data = body.data
+ expect(data).to.have.lengthOf(1)
+ expect(data[0].displayName).to.equal('my super playlist')
+ }
+ })
- expect(res.body.total).to.equal(1)
+ it('Should list account playlists', 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')
- }
- })
+ {
+ const body = await servers[1].playlists.listByAccount({ handle: 'root', start: 1, count: 2, sort: '-createdAt' })
+ expect(body.total).to.equal(2)
- it('Should list account playlists', async function () {
- this.timeout(30000)
+ const data = body.data
+ expect(data).to.have.lengthOf(1)
+ expect(data[0].displayName).to.equal('playlist 2')
+ }
- {
- const res = await getAccountPlaylistsList(servers[ 1 ].url, 'root', 1, 2, '-createdAt')
+ {
+ const body = await servers[1].playlists.listByAccount({ handle: 'root', start: 1, count: 2, sort: 'createdAt' })
+ expect(body.total).to.equal(2)
- expect(res.body.total).to.equal(2)
+ const data = body.data
+ expect(data).to.have.lengthOf(1)
+ expect(data[0].displayName).to.equal('playlist 3')
+ }
- const data: VideoPlaylist[] = res.body.data
- expect(data).to.have.lengthOf(1)
- expect(data[ 0 ].displayName).to.equal('playlist 2')
- }
+ {
+ const body = await servers[1].playlists.listByAccount({ handle: 'root', sort: 'createdAt', search: '3' })
+ expect(body.total).to.equal(1)
- {
- const res = await getAccountPlaylistsList(servers[ 1 ].url, 'root', 1, 2, 'createdAt')
+ const data = body.data
+ expect(data).to.have.lengthOf(1)
+ expect(data[0].displayName).to.equal('playlist 3')
+ }
- expect(res.body.total).to.equal(2)
+ {
+ const body = await servers[1].playlists.listByAccount({ handle: 'root', sort: 'createdAt', search: '4' })
+ expect(body.total).to.equal(0)
- const data: VideoPlaylist[] = res.body.data
- expect(data).to.have.lengthOf(1)
- expect(data[ 0 ].displayName).to.equal('playlist 3')
- }
+ const data = body.data
+ expect(data).to.have.lengthOf(0)
+ }
+ })
})
- it('Should not list unlisted or private playlists', async function () {
- this.timeout(30000)
+ describe('Playlist rights', function () {
+ let unlistedPlaylist: VideoPlaylistCreateResult
+ let privatePlaylist: VideoPlaylistCreateResult
+
+ before(async function () {
+ this.timeout(30000)
+
+ {
+ unlistedPlaylist = await servers[1].playlists.create({
+ attributes: {
+ displayName: 'playlist unlisted',
+ privacy: VideoPlaylistPrivacy.UNLISTED,
+ videoChannelId: servers[1].store.channel.id
+ }
+ })
+ }
- await createVideoPlaylist({
- url: servers[ 1 ].url,
- token: servers[ 1 ].accessToken,
- playlistAttrs: {
- displayName: 'playlist unlisted',
- privacy: VideoPlaylistPrivacy.UNLISTED
+ {
+ privatePlaylist = await servers[1].playlists.create({
+ attributes: {
+ displayName: 'playlist private',
+ privacy: VideoPlaylistPrivacy.PRIVATE
+ }
+ })
}
+
+ await waitJobs(servers)
+ await wait(3000)
})
- await createVideoPlaylist({
- url: servers[ 1 ].url,
- token: servers[ 1 ].accessToken,
- playlistAttrs: {
- displayName: 'playlist private',
- privacy: VideoPlaylistPrivacy.PRIVATE
+ it('Should not list unlisted or private playlists', async function () {
+ for (const server of servers) {
+ const results = [
+ await server.playlists.listByAccount({ handle: 'root@localhost:' + servers[1].port, sort: '-createdAt' }),
+ await server.playlists.list({ start: 0, count: 2, sort: '-createdAt' })
+ ]
+
+ expect(results[0].total).to.equal(2)
+ expect(results[1].total).to.equal(3)
+
+ for (const body of results) {
+ const data = 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)
-
- 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')
- ]
+ it('Should not get unlisted playlist using only the id', async function () {
+ await servers[1].playlists.get({ playlistId: unlistedPlaylist.id, expectedStatus: 404 })
+ })
- expect(results[0].body.total).to.equal(2)
- expect(results[1].body.total).to.equal(3)
+ it('Should get unlisted plyaylist using uuid or shortUUID', async function () {
+ await servers[1].playlists.get({ playlistId: unlistedPlaylist.uuid })
+ await servers[1].playlists.get({ playlistId: unlistedPlaylist.shortUUID })
+ })
- 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')
+ it('Should not get private playlist without token', async function () {
+ for (const id of [ privatePlaylist.id, privatePlaylist.uuid, privatePlaylist.shortUUID ]) {
+ await servers[1].playlists.get({ playlistId: id, expectedStatus: 401 })
}
- }
- })
-
- 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
+ it('Should get private playlist with a token', async function () {
+ for (const id of [ privatePlaylist.id, privatePlaylist.uuid, privatePlaylist.shortUUID ]) {
+ await servers[1].playlists.get({ token: servers[1].accessToken, playlistId: id })
+ }
})
+ })
- await waitJobs(servers)
+ describe('Update playlists', function () {
- for (const server of servers) {
- const res = await getVideoPlaylist(server.url, playlistServer2UUID2)
- const playlist: VideoPlaylist = res.body
+ it('Should update a playlist', async function () {
+ this.timeout(30000)
- expect(playlist.displayName).to.equal('playlist 3 updated')
- expect(playlist.description).to.equal('description updated')
+ await servers[1].playlists.update({
+ attributes: {
+ displayName: 'playlist 3 updated',
+ description: 'description updated',
+ privacy: VideoPlaylistPrivacy.UNLISTED,
+ thumbnailfile: 'thumbnail.jpg',
+ videoChannelId: servers[1].store.channel.id
+ },
+ playlistId: playlistServer2Id2
+ })
- 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')
+ for (const server of servers) {
+ const playlist = await server.playlists.get({ playlistId: playlistServer2UUID2 })
- expect(playlist.videosLength).to.equal(2)
+ expect(playlist.displayName).to.equal('playlist 3 updated')
+ expect(playlist.description).to.equal('description updated')
- 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(playlist.privacy.id).to.equal(VideoPlaylistPrivacy.UNLISTED)
+ expect(playlist.privacy.label).to.equal('Unlisted')
- it('Should create a playlist containing different startTimestamp/endTimestamp videos', async function () {
- this.timeout(30000)
+ expect(playlist.type.id).to.equal(VideoPlaylistType.REGULAR)
+ expect(playlist.type.label).to.equal('Regular')
- const addVideo = (elementAttrs: any) => {
- return addVideoInPlaylist({ url: servers[0].url, token: servers[0].accessToken, playlistId: playlistServer1Id, elementAttrs })
- }
+ expect(playlist.videosLength).to.equal(2)
- const res = await createVideoPlaylist({
- url: servers[ 0 ].url,
- token: servers[ 0 ].accessToken,
- playlistAttrs: {
- displayName: 'playlist 4',
- privacy: VideoPlaylistPrivacy.PUBLIC,
- videoChannelId: servers[0].videoChannel.id
+ 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')
}
})
+ })
- playlistServer1Id = res.body.videoPlaylist.id
- playlistServer1UUID = res.body.videoPlaylist.uuid
+ describe('Element timestamps', function () {
- 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 })
+ it('Should create a playlist containing different startTimestamp/endTimestamp videos', async function () {
+ this.timeout(30000)
- await waitJobs(servers)
- })
+ const addVideo = (attributes: any) => {
+ return commands[0].addElement({ playlistId: playlistServer1Id, attributes })
+ }
- it('Should correctly list playlist videos', async function () {
- this.timeout(30000)
+ const playlistDisplayName = 'playlist 4'
+ const playlist = await commands[0].create({
+ attributes: {
+ displayName: playlistDisplayName,
+ privacy: VideoPlaylistPrivacy.PUBLIC,
+ videoChannelId: servers[0].store.channel.id
+ }
+ })
- for (const server of servers) {
- const res = await getPlaylistVideos(server.url, server.accessToken, playlistServer1UUID, 0, 10)
+ playlistServer1Id = playlist.id
+ playlistServer1DisplayName = playlistDisplayName
+ playlistServer1UUID = playlist.uuid
+
+ await addVideo({ videoId: servers[0].store.videos[0].uuid, startTimestamp: 15, stopTimestamp: 28 })
+ await addVideo({ videoId: servers[2].store.videos[1].uuid, startTimestamp: 35 })
+ await addVideo({ videoId: servers[2].store.videos[2].uuid })
+ {
+ const element = await addVideo({ videoId: servers[0].store.videos[3].uuid, stopTimestamp: 35 })
+ playlistElementServer1Video4 = element.id
+ }
- expect(res.body.total).to.equal(6)
+ {
+ const element = await addVideo({ videoId: servers[0].store.videos[4].uuid, startTimestamp: 45, stopTimestamp: 60 })
+ playlistElementServer1Video5 = element.id
+ }
- const videos: Video[] = res.body.data
- expect(videos).to.have.lengthOf(6)
+ {
+ const element = await addVideo({ videoId: nsfwVideoServer1, startTimestamp: 5 })
+ playlistElementNSFW = element.id
- 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)
+ await addVideo({ videoId: nsfwVideoServer1, startTimestamp: 4 })
+ await addVideo({ videoId: nsfwVideoServer1 })
+ }
- 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 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
+ it('Should correctly list playlist videos', async function () {
+ this.timeout(30000)
- 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)
+ for (const server of servers) {
+ {
+ const body = await server.playlists.listVideos({ playlistId: playlistServer1UUID, start: 0, count: 10 })
+
+ expect(body.total).to.equal(8)
+
+ const videoElements = body.data
+ expect(videoElements).to.have.lengthOf(8)
+
+ 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)
+
+ 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
+
+ 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(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)
+
+ 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)
+
+ 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
+
+ expect(videoElements[6].video.name).to.equal('NSFW video')
+ expect(videoElements[6].position).to.equal(7)
+ expect(videoElements[6].startTimestamp).to.equal(4)
+ expect(videoElements[6].stopTimestamp).to.be.null
+
+ expect(videoElements[7].video.name).to.equal('NSFW video')
+ expect(videoElements[7].position).to.equal(8)
+ expect(videoElements[7].startTimestamp).to.be.null
+ expect(videoElements[7].stopTimestamp).to.be.null
+ }
- 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)
+ {
+ const body = await server.playlists.listVideos({ playlistId: playlistServer1UUID, start: 0, count: 2 })
+ expect(body.data).to.have.lengthOf(2)
+ }
+ }
+ })
+ })
- 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
+ describe('Element type', function () {
+ let groupUser1: PeerTubeServer[]
+ let groupWithoutToken1: PeerTubeServer[]
+ let group1: PeerTubeServer[]
+ let group2: PeerTubeServer[]
- 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
+ let video1: string
+ let video2: string
+ let video3: string
- const res3 = await getPlaylistVideos(server.url, server.accessToken, playlistServer1UUID, 0, 2)
- expect(res3.body.data).to.have.lengthOf(2)
- }
- })
+ before(async function () {
+ this.timeout(60000)
- it('Should reorder the playlist', async function () {
- this.timeout(30000)
+ groupUser1 = [ Object.assign({}, servers[0], { accessToken: userTokenServer1 }) ]
+ groupWithoutToken1 = [ Object.assign({}, servers[0], { accessToken: undefined }) ]
+ group1 = [ servers[0] ]
+ group2 = [ servers[1], servers[2] ]
- {
- await reorderVideosPlaylist({
- url: servers[ 0 ].url,
- token: servers[ 0 ].accessToken,
- playlistId: playlistServer1Id,
- elementAttrs: {
- startPosition: 2,
- insertAfterPosition: 3
+ const playlist = await commands[0].create({
+ token: userTokenServer1,
+ attributes: {
+ displayName: 'playlist 56',
+ privacy: VideoPlaylistPrivacy.PUBLIC,
+ videoChannelId: servers[0].store.channel.id
}
})
- await waitJobs(servers)
+ const playlistServer1Id2 = playlist.id
+ playlistServer1UUID2 = playlist.uuid
- 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)
-
- 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'
- ])
+ const addVideo = (attributes: any) => {
+ return commands[0].addElement({ token: userTokenServer1, playlistId: playlistServer1Id2, attributes })
}
- }
- {
- await reorderVideosPlaylist({
- url: servers[0].url,
- token: servers[0].accessToken,
- playlistId: playlistServer1Id,
- elementAttrs: {
- startPosition: 1,
- reorderLength: 3,
- insertAfterPosition: 4
- }
- })
+ video1 = (await servers[0].videos.quickUpload({ name: 'video 89', token: userTokenServer1 })).uuid
+ video2 = (await servers[1].videos.quickUpload({ name: 'video 90' })).uuid
+ video3 = (await servers[0].videos.quickUpload({ name: 'video 91', nsfw: true })).uuid
await waitJobs(servers)
- 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)
-
- 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 reorderVideosPlaylist({
- url: servers[0].url,
- token: servers[0].accessToken,
- playlistId: playlistServer1Id,
- elementAttrs: {
- startPosition: 6,
- insertAfterPosition: 3
- }
- })
+ await addVideo({ videoId: video1, startTimestamp: 15, stopTimestamp: 28 })
+ await addVideo({ videoId: video2, startTimestamp: 35 })
+ await addVideo({ videoId: video3 })
await waitJobs(servers)
+ })
- for (const server of servers) {
- const res = await getPlaylistVideos(server.url, server.accessToken, playlistServer1UUID, 0, 10)
- const videos: Video[] = res.body.data
-
- const names = videos.map(v => v.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 <= videos.length; i++) {
- expect(videos[i - 1].playlistElement.position).to.equal(i)
- }
+ it('Should update the element type if the video is private', async function () {
+ this.timeout(20000)
+
+ const name = 'video 89'
+ const position = 1
+
+ {
+ await servers[0].videos.update({ id: video1, attributes: { privacy: VideoPrivacy.PRIVATE } })
+ await waitJobs(servers)
+
+ 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)
}
- }
- })
- it('Should update startTimestamp/endTimestamp of some elements', async function () {
- this.timeout(30000)
+ {
+ await servers[0].videos.update({ id: video1, attributes: { privacy: VideoPrivacy.PUBLIC } })
+ await waitJobs(servers)
- await updateVideoPlaylistElement({
- url: servers[0].url,
- token: servers[0].accessToken,
- playlistId: playlistServer1Id,
- videoId: servers[0].videos[3].uuid,
- elementAttrs: {
- startTimestamp: 1
+ 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)
}
})
- await updateVideoPlaylistElement({
- url: servers[0].url,
- token: servers[0].accessToken,
- playlistId: playlistServer1Id,
- videoId: servers[0].videos[4].uuid,
- elementAttrs: {
- stopTimestamp: null
+ it('Should update the element type if the video is blacklisted', async function () {
+ this.timeout(20000)
+
+ const name = 'video 89'
+ const position = 1
+
+ {
+ await servers[0].blacklist.add({ videoId: video1, reason: 'reason', unfederate: true })
+ await waitJobs(servers)
+
+ 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)
+ }
+
+ {
+ await servers[0].blacklist.remove({ videoId: video1 })
+ await waitJobs(servers)
+
+ 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)
}
})
- await waitJobs(servers)
+ it('Should update the element type if the account or server of the video is blocked', async function () {
+ this.timeout(90000)
- for (const server of servers) {
- const res = await getPlaylistVideos(server.url, server.accessToken, playlistServer1UUID, 0, 10)
- const videos: Video[] = res.body.data
-
- 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)
-
- 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
- }
- })
+ const command = servers[0].blocklist
- 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 name = 'video 90'
+ const position = 2
- {
- 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)
- }
+ {
+ await command.addToMyBlocklist({ token: userTokenServer1, account: 'root@localhost:' + servers[1].port })
+ await waitJobs(servers)
- {
- 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)
- }
+ await checkPlaylistElementType(groupUser1, playlistServer1UUID2, VideoPlaylistElementType.UNAVAILABLE, position, name, 3)
+ await checkPlaylistElementType(group2, playlistServer1UUID2, VideoPlaylistElementType.REGULAR, position, name, 3)
- {
- 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)
- }
+ await command.removeFromMyBlocklist({ token: userTokenServer1, account: 'root@localhost:' + servers[1].port })
+ await waitJobs(servers)
- expect(obj[42000]).to.have.lengthOf(0)
- expect(obj[43000]).to.have.lengthOf(0)
- })
+ await checkPlaylistElementType(group2, playlistServer1UUID2, VideoPlaylistElementType.REGULAR, position, name, 3)
+ }
- it('Should automatically update updatedAt field of playlists', async function () {
- const server = servers[1]
- const videoId = servers[1].videos[5].id
+ {
+ await command.addToMyBlocklist({ token: userTokenServer1, server: 'localhost:' + servers[1].port })
+ await waitJobs(servers)
- async function getPlaylistNames () {
- const res = await getAccountPlaylistsListWithToken(server.url, server.accessToken, 'root', 0, 5, undefined, '-updatedAt')
+ await checkPlaylistElementType(groupUser1, playlistServer1UUID2, VideoPlaylistElementType.UNAVAILABLE, position, name, 3)
+ await checkPlaylistElementType(group2, playlistServer1UUID2, VideoPlaylistElementType.REGULAR, position, name, 3)
- return (res.body.data as VideoPlaylist[]).map(p => p.displayName)
- }
+ await command.removeFromMyBlocklist({ token: userTokenServer1, server: 'localhost:' + servers[1].port })
+ await waitJobs(servers)
- 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 })
+ await checkPlaylistElementType(group2, playlistServer1UUID2, VideoPlaylistElementType.REGULAR, position, name, 3)
+ }
- const names1 = await getPlaylistNames()
- expect(names1[0]).to.equal('playlist 3 updated')
- expect(names1[1]).to.equal('playlist 2')
+ {
+ await command.addToServerBlocklist({ account: 'root@localhost:' + servers[1].port })
+ await waitJobs(servers)
- await removeVideoFromPlaylist({ url: server.url, token: server.accessToken, playlistId: playlistServer2Id1, videoId })
+ await checkPlaylistElementType(groupUser1, playlistServer1UUID2, VideoPlaylistElementType.UNAVAILABLE, position, name, 3)
+ await checkPlaylistElementType(group2, playlistServer1UUID2, VideoPlaylistElementType.REGULAR, position, name, 3)
- const names2 = await getPlaylistNames()
- expect(names2[0]).to.equal('playlist 2')
- expect(names2[1]).to.equal('playlist 3 updated')
+ await command.removeFromServerBlocklist({ account: 'root@localhost:' + servers[1].port })
+ await waitJobs(servers)
- await removeVideoFromPlaylist({ url: server.url, token: server.accessToken, playlistId: playlistServer2Id2, videoId })
+ await checkPlaylistElementType(group2, playlistServer1UUID2, VideoPlaylistElementType.REGULAR, position, name, 3)
+ }
- const names3 = await getPlaylistNames()
- expect(names3[0]).to.equal('playlist 3 updated')
- expect(names3[1]).to.equal('playlist 2')
- })
+ {
+ await command.addToServerBlocklist({ server: 'localhost:' + servers[1].port })
+ await waitJobs(servers)
- it('Should delete some elements', async function () {
- this.timeout(30000)
+ await checkPlaylistElementType(groupUser1, playlistServer1UUID2, VideoPlaylistElementType.UNAVAILABLE, position, name, 3)
+ await checkPlaylistElementType(group2, playlistServer1UUID2, VideoPlaylistElementType.REGULAR, position, name, 3)
- await removeVideoFromPlaylist({
- url: servers[0].url,
- token: servers[0].accessToken,
- playlistId: playlistServer1Id,
- videoId: servers[0].videos[3].uuid
+ await command.removeFromServerBlocklist({ server: 'localhost:' + servers[1].port })
+ await waitJobs(servers)
+
+ await checkPlaylistElementType(group2, playlistServer1UUID2, VideoPlaylistElementType.REGULAR, position, name, 3)
+ }
})
- await removeVideoFromPlaylist({
- url: servers[0].url,
- token: servers[0].accessToken,
- playlistId: playlistServer1Id,
- videoId: nsfwVideoServer1
+ it('Should hide the video if it is NSFW', async function () {
+ const body = await commands[0].listVideos({ token: userTokenServer1, playlistId: playlistServer1UUID2, query: { nsfw: 'false' } })
+ expect(body.total).to.equal(3)
+
+ const elements = body.data
+ const element = elements.find(e => e.position === 3)
+
+ expect(element).to.exist
+ expect(element.video).to.be.null
+ expect(element.type).to.equal(VideoPlaylistElementType.UNAVAILABLE)
})
- await waitJobs(servers)
+ })
- for (const server of servers) {
- const res = await getPlaylistVideos(server.url, server.accessToken, playlistServer1UUID, 0, 10)
+ describe('Managing playlist elements', function () {
+
+ it('Should reorder the playlist', async function () {
+ this.timeout(30000)
+
+ {
+ await commands[0].reorderElements({
+ playlistId: playlistServer1Id,
+ attributes: {
+ startPosition: 2,
+ insertAfterPosition: 3
+ }
+ })
+
+ await waitJobs(servers)
+
+ for (const server of servers) {
+ const body = await server.playlists.listVideos({ playlistId: playlistServer1UUID, start: 0, count: 10 })
+ const names = body.data.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',
+ 'NSFW video',
+ 'NSFW video'
+ ])
+ }
+ }
+
+ {
+ await commands[0].reorderElements({
+ playlistId: playlistServer1Id,
+ attributes: {
+ startPosition: 1,
+ reorderLength: 3,
+ insertAfterPosition: 4
+ }
+ })
+
+ await waitJobs(servers)
+
+ for (const server of servers) {
+ const body = await server.playlists.listVideos({ playlistId: playlistServer1UUID, start: 0, count: 10 })
+ const names = body.data.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',
+ 'NSFW video',
+ 'NSFW video'
+ ])
+ }
+ }
- expect(res.body.total).to.equal(4)
+ {
+ await commands[0].reorderElements({
+ playlistId: playlistServer1Id,
+ attributes: {
+ startPosition: 6,
+ insertAfterPosition: 3
+ }
+ })
+
+ await waitJobs(servers)
+
+ for (const server of servers) {
+ const { data: elements } = await server.playlists.listVideos({ playlistId: playlistServer1UUID, start: 0, count: 10 })
+ 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',
+ 'NSFW video',
+ 'NSFW video'
+ ])
+
+ for (let i = 1; i <= elements.length; i++) {
+ expect(elements[i - 1].position).to.equal(i)
+ }
+ }
+ }
+ })
- const videos: Video[] = res.body.data
- expect(videos).to.have.lengthOf(4)
+ it('Should update startTimestamp/endTimestamp of some elements', async function () {
+ this.timeout(30000)
- expect(videos[ 0 ].name).to.equal('video 0 server 1')
- expect(videos[ 0 ].playlistElement.position).to.equal(1)
+ await commands[0].updateElement({
+ playlistId: playlistServer1Id,
+ elementId: playlistElementServer1Video4,
+ attributes: {
+ startTimestamp: 1
+ }
+ })
- expect(videos[ 1 ].name).to.equal('video 2 server 3')
- expect(videos[ 1 ].playlistElement.position).to.equal(2)
+ await commands[0].updateElement({
+ playlistId: playlistServer1Id,
+ elementId: playlistElementServer1Video5,
+ attributes: {
+ stopTimestamp: null
+ }
+ })
- expect(videos[ 2 ].name).to.equal('video 1 server 3')
- expect(videos[ 2 ].playlistElement.position).to.equal(3)
+ await waitJobs(servers)
- expect(videos[ 3 ].name).to.equal('video 4 server 1')
- expect(videos[ 3 ].playlistElement.position).to.equal(4)
- }
- })
+ for (const server of servers) {
+ const { data: elements } = await server.playlists.listVideos({ playlistId: playlistServer1UUID, start: 0, count: 10 })
- it('Should be able to create a public playlist, and set it to private', async function () {
- this.timeout(30000)
+ 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 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[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 videoPlaylistIds = res.body.videoPlaylist
- await waitJobs(servers)
+ it('Should check videos existence in my playlist', async function () {
+ const videoIds = [
+ servers[0].store.videos[0].id,
+ 42000,
+ servers[0].store.videos[3].id,
+ 43000,
+ servers[0].store.videos[4].id
+ ]
+ const obj = await commands[0].videosExist({ videoIds })
+
+ {
+ const elem = obj[servers[0].store.videos[0].id]
+ expect(elem).to.have.lengthOf(1)
+ expect(elem[0].playlistElementId).to.exist
+ expect(elem[0].playlistDisplayName).to.equal(playlistServer1DisplayName)
+ expect(elem[0].playlistShortUUID).to.equal(uuidToShort(playlistServer1UUID))
+ expect(elem[0].playlistId).to.equal(playlistServer1Id)
+ expect(elem[0].startTimestamp).to.equal(15)
+ expect(elem[0].stopTimestamp).to.equal(28)
+ }
- for (const server of servers) {
- await getVideoPlaylist(server.url, videoPlaylistIds.uuid, 200)
- }
+ {
+ const elem = obj[servers[0].store.videos[3].id]
+ expect(elem).to.have.lengthOf(1)
+ expect(elem[0].playlistElementId).to.equal(playlistElementServer1Video4)
+ expect(elem[0].playlistDisplayName).to.equal(playlistServer1DisplayName)
+ expect(elem[0].playlistShortUUID).to.equal(uuidToShort(playlistServer1UUID))
+ expect(elem[0].playlistId).to.equal(playlistServer1Id)
+ expect(elem[0].startTimestamp).to.equal(1)
+ expect(elem[0].stopTimestamp).to.equal(35)
+ }
- const playlistAttrs = { privacy: VideoPlaylistPrivacy.PRIVATE }
- await updateVideoPlaylist({ url: servers[0].url, token: servers[0].accessToken, playlistId: videoPlaylistIds.id, playlistAttrs })
+ {
+ const elem = obj[servers[0].store.videos[4].id]
+ expect(elem).to.have.lengthOf(1)
+ expect(elem[0].playlistId).to.equal(playlistServer1Id)
+ expect(elem[0].playlistDisplayName).to.equal(playlistServer1DisplayName)
+ expect(elem[0].playlistShortUUID).to.equal(uuidToShort(playlistServer1UUID))
+ expect(elem[0].startTimestamp).to.equal(45)
+ expect(elem[0].stopTimestamp).to.equal(null)
+ }
- await waitJobs(servers)
+ expect(obj[42000]).to.have.lengthOf(0)
+ expect(obj[43000]).to.have.lengthOf(0)
+ })
- for (const server of [ servers[1], servers[2] ]) {
- await getVideoPlaylist(server.url, videoPlaylistIds.uuid, 404)
- }
- await getVideoPlaylist(servers[0].url, videoPlaylistIds.uuid, 401)
+ it('Should automatically update updatedAt field of playlists', async function () {
+ const server = servers[1]
+ const videoId = servers[1].store.videos[5].id
- await getVideoPlaylistWithToken(servers[0].url, servers[0].accessToken, videoPlaylistIds.uuid, 200)
- })
+ async function getPlaylistNames () {
+ const { data } = await server.playlists.listByAccount({ token: server.accessToken, handle: 'root', sort: '-updatedAt' })
- it('Should delete the playlist on server 1 and delete on server 2 and 3', async function () {
- this.timeout(30000)
+ return data.map(p => p.displayName)
+ }
- await deleteVideoPlaylist(servers[0].url, servers[0].accessToken, playlistServer1Id)
+ const attributes = { videoId }
+ const element1 = await server.playlists.addElement({ playlistId: playlistServer2Id1, attributes })
+ const element2 = await server.playlists.addElement({ playlistId: playlistServer2Id2, attributes })
- await waitJobs(servers)
+ const names1 = await getPlaylistNames()
+ expect(names1[0]).to.equal('playlist 3 updated')
+ expect(names1[1]).to.equal('playlist 2')
- for (const server of servers) {
- await getVideoPlaylist(server.url, playlistServer1UUID, 404)
- }
- })
+ await server.playlists.removeElement({ playlistId: playlistServer2Id1, elementId: element1.id })
- it('Should have deleted the thumbnail on server 1, 2 and 3', async function () {
- this.timeout(30000)
+ const names2 = await getPlaylistNames()
+ expect(names2[0]).to.equal('playlist 2')
+ expect(names2[1]).to.equal('playlist 3 updated')
- for (const server of servers) {
- await checkPlaylistFilesWereRemoved(playlistServer1UUID, server.internalServerNumber)
- }
- })
+ await server.playlists.removeElement({ playlistId: playlistServer2Id2, elementId: element2.id })
- it('Should unfollow servers 1 and 2 and hide their playlists', async function () {
- this.timeout(30000)
+ const names3 = await getPlaylistNames()
+ expect(names3[0]).to.equal('playlist 3 updated')
+ expect(names3[1]).to.equal('playlist 2')
+ })
- const finder = data => data.find(p => p.displayName === 'my super playlist')
+ it('Should delete some elements', 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
- }
+ await commands[0].removeElement({ playlistId: playlistServer1Id, elementId: playlistElementServer1Video4 })
+ await commands[0].removeElement({ playlistId: playlistServer1Id, elementId: playlistElementNSFW })
- await unfollow(servers[2].url, servers[2].accessToken, servers[0])
+ await waitJobs(servers)
- {
- const res = await getVideoPlaylistsList(servers[ 2 ].url, 0, 5)
- expect(res.body.total).to.equal(1)
+ for (const server of servers) {
+ const body = await server.playlists.listVideos({ playlistId: playlistServer1UUID, start: 0, count: 10 })
+ expect(body.total).to.equal(6)
- expect(finder(res.body.data)).to.be.undefined
- }
- })
+ const elements = body.data
+ expect(elements).to.have.lengthOf(6)
+
+ expect(elements[0].video.name).to.equal('video 0 server 1')
+ expect(elements[0].position).to.equal(1)
+
+ expect(elements[1].video.name).to.equal('video 2 server 3')
+ expect(elements[1].position).to.equal(2)
- it('Should delete a channel and put the associated playlist in private mode', 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 addVideoChannel(servers[0].url, servers[0].accessToken, { name: 'super_channel', displayName: 'super channel' })
- const videoChannelId = res.body.videoChannel.id
+ expect(elements[3].video.name).to.equal('video 4 server 1')
+ expect(elements[3].position).to.equal(4)
- const res2 = await createVideoPlaylist({
- url: servers[0].url,
- token: servers[0].accessToken,
- playlistAttrs: {
- displayName: 'channel playlist',
- privacy: VideoPlaylistPrivacy.PUBLIC,
- videoChannelId
+ expect(elements[4].video.name).to.equal('NSFW video')
+ expect(elements[4].position).to.equal(5)
+
+ expect(elements[5].video.name).to.equal('NSFW video')
+ expect(elements[5].position).to.equal(6)
}
})
- const videoPlaylistUUID = res2.body.videoPlaylist.uuid
- await waitJobs(servers)
+ it('Should be able to create a public playlist, and set it to private', async function () {
+ this.timeout(30000)
- await deleteVideoChannel(servers[0].url, servers[0].accessToken, 'super_channel')
+ const videoPlaylistIds = await commands[0].create({
+ attributes: {
+ displayName: 'my super public playlist',
+ privacy: VideoPlaylistPrivacy.PUBLIC,
+ videoChannelId: servers[0].store.channel.id
+ }
+ })
- await waitJobs(servers)
+ 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)
+ for (const server of servers) {
+ await server.playlists.get({ playlistId: videoPlaylistIds.uuid, expectedStatus: HttpStatusCode.OK_200 })
+ }
+
+ const attributes = { privacy: VideoPlaylistPrivacy.PRIVATE }
+ await commands[0].update({ playlistId: videoPlaylistIds.id, attributes })
+
+ await waitJobs(servers)
- await getVideoPlaylist(servers[1].url, videoPlaylistUUID, 404)
+ for (const server of [ servers[1], servers[2] ]) {
+ await server.playlists.get({ playlistId: videoPlaylistIds.uuid, expectedStatus: HttpStatusCode.NOT_FOUND_404 })
+ }
+
+ await commands[0].get({ playlistId: videoPlaylistIds.uuid, expectedStatus: HttpStatusCode.UNAUTHORIZED_401 })
+ await commands[0].get({ token: servers[0].accessToken, playlistId: videoPlaylistIds.uuid, expectedStatus: HttpStatusCode.OK_200 })
+ })
})
- it('Should delete an account and delete its playlists', async function () {
- this.timeout(30000)
+ describe('Playlist deletion', function () {
+
+ it('Should delete the playlist on server 1 and delete on server 2 and 3', async function () {
+ this.timeout(30000)
+
+ await commands[0].delete({ playlistId: playlistServer1Id })
- 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 waitJobs(servers)
+
+ for (const server of servers) {
+ await server.playlists.get({ playlistId: playlistServer1UUID, expectedStatus: HttpStatusCode.NOT_FOUND_404 })
+ }
+ })
+
+ it('Should have deleted the thumbnail on server 1, 2 and 3', async function () {
+ this.timeout(30000)
+
+ for (const server of servers) {
+ await checkPlaylistFilesWereRemoved(playlistServer1UUID, server)
+ }
})
- const userId = res.body.user.id
- const userAccessToken = await userLogin(servers[0], user)
+ it('Should unfollow servers 1 and 2 and hide their playlists', async function () {
+ this.timeout(30000)
+
+ const finder = (data: VideoPlaylist[]) => data.find(p => p.displayName === 'my super playlist')
- const resChannel = await getMyUserInformation(servers[0].url, userAccessToken)
- const userChannel = (resChannel.body as User).videoChannels[0]
+ {
+ const body = await servers[2].playlists.list({ start: 0, count: 5 })
+ expect(body.total).to.equal(3)
- await createVideoPlaylist({
- url: servers[0].url,
- token: userAccessToken,
- playlistAttrs: {
- displayName: 'playlist to be deleted',
- privacy: VideoPlaylistPrivacy.PUBLIC,
- videoChannelId: userChannel.id
+ expect(finder(body.data)).to.not.be.undefined
+ }
+
+ await servers[2].follows.unfollow({ target: servers[0] })
+
+ {
+ const body = await servers[2].playlists.list({ start: 0, count: 5 })
+ expect(body.total).to.equal(1)
+
+ expect(finder(body.data)).to.be.undefined
}
})
- await waitJobs(servers)
+ it('Should delete a channel and put the associated playlist in private mode', async function () {
+ this.timeout(30000)
- const finder = data => data.find(p => p.displayName === 'playlist to be deleted')
+ const channel = await servers[0].channels.create({ attributes: { name: 'super_channel', displayName: 'super channel' } })
- {
- 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
+ const playlistCreated = await commands[0].create({
+ attributes: {
+ displayName: 'channel playlist',
+ privacy: VideoPlaylistPrivacy.PUBLIC,
+ videoChannelId: channel.id
+ }
+ })
+
+ await waitJobs(servers)
+
+ await servers[0].channels.delete({ channelName: 'super_channel' })
+
+ await waitJobs(servers)
+
+ const body = await commands[0].get({ token: servers[0].accessToken, playlistId: playlistCreated.uuid })
+ expect(body.displayName).to.equal('channel playlist')
+ expect(body.privacy.id).to.equal(VideoPlaylistPrivacy.PRIVATE)
+
+ await servers[1].playlists.get({ playlistId: playlistCreated.uuid, expectedStatus: HttpStatusCode.NOT_FOUND_404 })
+ })
+
+ it('Should delete an account and delete its playlists', async function () {
+ this.timeout(30000)
+
+ const { userId, token } = await servers[0].users.generate('user_1')
+
+ const { videoChannels } = await servers[0].users.getMyInfo({ token })
+ const userChannel = videoChannels[0]
+
+ await commands[0].create({
+ attributes: {
+ displayName: 'playlist to be deleted',
+ privacy: VideoPlaylistPrivacy.PUBLIC,
+ videoChannelId: userChannel.id
+ }
+ })
+
+ await waitJobs(servers)
+
+ const finder = (data: VideoPlaylist[]) => data.find(p => p.displayName === 'playlist to be deleted')
+
+ {
+ for (const server of [ servers[0], servers[1] ]) {
+ const body = await server.playlists.list({ start: 0, count: 15 })
+
+ expect(finder(body.data)).to.not.be.undefined
+ }
}
- }
- await removeUser(servers[0].url, userId, servers[0].accessToken)
- await waitJobs(servers)
+ await servers[0].users.remove({ userId })
+ 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 body = await server.playlists.list({ start: 0, count: 15 })
+
+ expect(finder(body.data)).to.be.undefined
+ }
}
- }
+ })
})
after(async function () {