]> git.immae.eu Git - github/Chocobozzz/PeerTube.git/blobdiff - server/tests/api/videos/video-playlists.ts
Introduce blocklist command
[github/Chocobozzz/PeerTube.git] / server / tests / api / videos / video-playlists.ts
index 8c3542906da2a1e7585db483ecebfaa1938fa0ac..90189721a6de0e57b472c31fa1a965c483f5661f 100644 (file)
@@ -1,51 +1,87 @@
-/* tslint:disable:no-unused-expression */
+/* eslint-disable @typescript-eslint/no-unused-expressions,@typescript-eslint/require-await */
 
-import * as chai from 'chai'
 import 'mocha'
+import * as chai from 'chai'
+import { HttpStatusCode } from '@shared/core-utils'
 import {
   addVideoChannel,
   addVideoInPlaylist,
+  addVideoToBlacklist,
   checkPlaylistFilesWereRemoved,
+  cleanupTests,
   createUser,
   createVideoPlaylist,
   deleteVideoChannel,
   deleteVideoPlaylist,
-  doubleFollow, doVideosExistInMyPlaylist,
+  doubleFollow,
+  doVideosExistInMyPlaylist,
   flushAndRunMultipleServers,
-  flushTests,
+  generateUserAccessToken,
+  getAccessToken,
   getAccountPlaylistsList,
-  getAccountPlaylistsListWithToken, getMyUserInformation,
+  getAccountPlaylistsListWithToken,
+  getMyUserInformation,
   getPlaylistVideos,
   getVideoChannelPlaylistsList,
   getVideoPlaylist,
   getVideoPlaylistPrivacies,
   getVideoPlaylistsList,
   getVideoPlaylistWithToken,
-  killallServers,
   removeUser,
+  removeVideoFromBlacklist,
   removeVideoFromPlaylist,
   reorderVideosPlaylist,
   ServerInfo,
   setAccessTokensToServers,
   setDefaultVideoChannel,
   testImage,
-  unfollow,
+  updateVideo,
   updateVideoPlaylist,
   updateVideoPlaylistElement,
   uploadVideo,
   uploadVideoAndGetId,
   userLogin,
+  wait,
   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 { 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'
+} from '@shared/extra-utils'
+import {
+  User,
+  VideoExistInPlaylist,
+  VideoPlaylist,
+  VideoPlaylistCreateResult,
+  VideoPlaylistElement,
+  VideoPlaylistElementType,
+  VideoPlaylistPrivacy,
+  VideoPlaylistType,
+  VideoPrivacy
+} from '@shared/models'
 
 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[] = []
 
@@ -55,9 +91,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)
 
@@ -73,799 +116,1147 @@ describe('Test video playlists', function () {
     await doubleFollow(servers[0], servers[2])
 
     {
-      const serverPromises: Promise<any>[][] = []
+      servers[0].videos = []
+      servers[1].videos = []
+      servers[2].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 resVideo = await uploadVideo(server.url, server.accessToken, { name, nsfw: false })
 
-        serverPromises.push(videoPromises)
+          server.videos.push(resVideo.body.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 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)
-
-    const privacies = res.body
-    expect(Object.keys(privacies)).to.have.length.at.least(3)
-
-    expect(privacies[3]).to.equal('Private')
-  })
+  describe('Get default playlists', function () {
+    it('Should list video playlist privacies', async function () {
+      const res = await getVideoPlaylistPrivacies(servers[0].url)
 
-  it('Should list watch later playlist', async function () {
-    const url = servers[ 0 ].url
-    const accessToken = servers[ 0 ].accessToken
+      const privacies = res.body
+      expect(Object.keys(privacies)).to.have.length.at.least(3)
 
-    {
-      const res = await getAccountPlaylistsListWithToken(url, accessToken, 'root', 0, 5, VideoPlaylistType.WATCH_LATER)
+      expect(privacies[3]).to.equal('Private')
+    })
 
-      expect(res.body.total).to.equal(1)
-      expect(res.body.data).to.have.lengthOf(1)
+    it('Should list watch later playlist', async function () {
+      const url = servers[0].url
+      const accessToken = servers[0].accessToken
 
-      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.WATCH_LATER)
 
-    {
-      const res = await getAccountPlaylistsListWithToken(url, accessToken, 'root', 0, 5, VideoPlaylistType.REGULAR)
+        expect(res.body.total).to.equal(1)
+        expect(res.body.data).to.have.lengthOf(1)
 
-      expect(res.body.total).to.equal(0)
-      expect(res.body.data).to.have.lengthOf(0)
-    }
+        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 getAccountPlaylistsList(url, 'root', 0, 5)
-      expect(res.body.total).to.equal(0)
-      expect(res.body.data).to.have.lengthOf(0)
-    }
-  })
+      {
+        const res = await getAccountPlaylistsListWithToken(url, accessToken, 'root', 0, 5, VideoPlaylistType.REGULAR)
 
-  it('Should create a playlist on server 1 and have the playlist on server 2 and 3', async function () {
-    this.timeout(30000)
+        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
+      {
+        const res = await getAccountPlaylistsList(url, 'root', 0, 5)
+        expect(res.body.total).to.equal(0)
+        expect(res.body.data).to.have.lengthOf(0)
       }
     })
 
-    await waitJobs(servers)
+    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)
 
-    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)
-
-        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 playlistId = res.body.data[0].id
+      await getPlaylistVideos(servers[0].url, token, playlistId, 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)
+  describe('Create and federate playlists', function () {
 
-    {
-      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
-    }
+    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 createVideoPlaylist({
-        url: servers[ 1 ].url,
-        token: servers[ 1 ].accessToken,
+      await createVideoPlaylist({
+        url: servers[0].url,
+        token: servers[0].accessToken,
         playlistAttrs: {
-          displayName: 'playlist 3',
+          displayName: 'my super playlist',
           privacy: VideoPlaylistPrivacy.PUBLIC,
+          description: 'my super description',
           thumbnailfile: 'thumbnail.jpg',
-          videoChannelId: servers[1].videoChannel.id
+          videoChannelId: servers[0].videoChannel.id
         }
       })
 
-      playlistServer2Id2 = res.body.videoPlaylist.id
-      playlistServer2UUID2 = res.body.videoPlaylist.uuid
-    }
+      await waitJobs(servers)
+      // Processing a playlist by the receiver could be long
+      await wait(3000)
 
-    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 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)
+        const playlistFromList = res.body.data[0] as VideoPlaylist
 
-    for (const server of [ servers[0], servers[1] ]) {
-      const res = await getVideoPlaylistsList(server.url, 0, 5)
+        const res2 = await getVideoPlaylist(server.url, playlistFromList.uuid)
+        const playlistFromGet = res2.body as VideoPlaylist
 
-      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)
+        for (const playlist of [ playlistFromGet, playlistFromList ]) {
+          expect(playlist.id).to.be.a('number')
+          expect(playlist.uuid).to.be.a('string')
 
-      const playlist3 = res.body.data.find(p => p.displayName === 'playlist 3')
-      expect(playlist3).to.not.be.undefined
-      await testImage(server.url, 'thumbnail', playlist3.thumbnailPath)
-    }
+          expect(playlist.isLocal).to.equal(server.serverNumber === 1)
 
-    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
-  })
+          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)
 
-  it('Should have the playlist on server 3 after a new follow', async function () {
-    this.timeout(30000)
+          expect(playlist.videosLength).to.equal(0)
 
-    // Server 2 and server 3 follow each other
-    await doubleFollow(servers[1], servers[2])
-
-    const res = await getVideoPlaylistsList(servers[2].url, 0, 5)
+          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 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)
+    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.data.find(p => p.displayName === 'playlist 3')).to.not.be.undefined
-  })
+      {
+        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
+      }
 
-  it('Should correctly list the playlists', async function () {
-    this.timeout(30000)
+      for (const 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 }
+        })
+      }
 
-    {
-      const res = await getVideoPlaylistsList(servers[ 2 ].url, 1, 2, 'createdAt')
+      await waitJobs(servers)
+      await wait(3000)
 
-      expect(res.body.total).to.equal(3)
+      for (const server of [ servers[0], servers[1] ]) {
+        const res = await getVideoPlaylistsList(server.url, 0, 5)
 
-      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 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 res = await getVideoPlaylistsList(servers[ 2 ].url, 1, 2, '-createdAt')
+        const playlist3 = res.body.data.find(p => p.displayName === 'playlist 3')
+        expect(playlist3).to.not.be.undefined
+        await testImage(server.url, 'thumbnail', playlist3.thumbnailPath)
+      }
 
-      expect(res.body.total).to.equal(3)
+      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 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 have the playlist on server 3 after a new follow', async function () {
+      this.timeout(30000)
 
-  it('Should list video channel playlists', async function () {
-    this.timeout(30000)
+      // Server 2 and server 3 follow each other
+      await doubleFollow(servers[1], servers[2])
 
-    {
-      const res = await getVideoChannelPlaylistsList(servers[ 0 ].url, 'root_channel', 0, 2, '-createdAt')
+      const res = await getVideoPlaylistsList(servers[2].url, 0, 5)
 
-      expect(res.body.total).to.equal(1)
+      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 data: VideoPlaylist[] = res.body.data
-      expect(data).to.have.lengthOf(1)
-      expect(data[ 0 ].displayName).to.equal('my super playlist')
-    }
+      expect(res.body.data.find(p => p.displayName === 'playlist 3')).to.not.be.undefined
+    })
   })
 
-  it('Should list account playlists', async function () {
-    this.timeout(30000)
+  describe('List playlists', function () {
 
-    {
-      const res = await getAccountPlaylistsList(servers[ 1 ].url, 'root', 1, 2, '-createdAt')
+    it('Should correctly list the playlists', async function () {
+      this.timeout(30000)
 
-      expect(res.body.total).to.equal(2)
+      {
+        const res = await getVideoPlaylistsList(servers[2].url, 1, 2, 'createdAt')
 
-      const data: VideoPlaylist[] = res.body.data
-      expect(data).to.have.lengthOf(1)
-      expect(data[ 0 ].displayName).to.equal('playlist 2')
-    }
+        expect(res.body.total).to.equal(3)
 
-    {
-      const res = await getAccountPlaylistsList(servers[ 1 ].url, 'root', 1, 2, 'createdAt')
-
-      expect(res.body.total).to.equal(2)
+        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: VideoPlaylist[] = res.body.data
-      expect(data).to.have.lengthOf(1)
-      expect(data[ 0 ].displayName).to.equal('playlist 3')
-    }
-  })
+      {
+        const res = await getVideoPlaylistsList(servers[2].url, 1, 2, '-createdAt')
 
-  it('Should not list unlisted or private playlists', async function () {
-    this.timeout(30000)
+        expect(res.body.total).to.equal(3)
 
-    await createVideoPlaylist({
-      url: servers[ 1 ].url,
-      token: servers[ 1 ].accessToken,
-      playlistAttrs: {
-        displayName: 'playlist unlisted',
-        privacy: VideoPlaylistPrivacy.UNLISTED
+        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 createVideoPlaylist({
-      url: servers[ 1 ].url,
-      token: servers[ 1 ].accessToken,
-      playlistAttrs: {
-        displayName: 'playlist private',
-        privacy: VideoPlaylistPrivacy.PRIVATE
+    it('Should list video channel playlists', async function () {
+      this.timeout(30000)
+
+      {
+        const res = await getVideoChannelPlaylistsList(servers[0].url, 'root_channel', 0, 2, '-createdAt')
+
+        expect(res.body.total).to.equal(1)
+
+        const data: VideoPlaylist[] = res.body.data
+        expect(data).to.have.lengthOf(1)
+        expect(data[0].displayName).to.equal('my super playlist')
       }
     })
 
-    await waitJobs(servers)
+    it('Should list account playlists', async function () {
+      this.timeout(30000)
 
-    for (const server of servers) {
-      const results = [
-        await getAccountPlaylistsList(server.url, 'root@localhost:9002', 0, 5, '-createdAt'),
-        await getVideoPlaylistsList(server.url, 0, 2, '-createdAt')
-      ]
+      {
+        const res = await getAccountPlaylistsList(servers[1].url, 'root', 1, 2, '-createdAt')
 
-      expect(results[0].body.total).to.equal(2)
-      expect(results[1].body.total).to.equal(3)
+        expect(res.body.total).to.equal(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')
+        expect(data).to.have.lengthOf(1)
+        expect(data[0].displayName).to.equal('playlist 2')
       }
-    }
-  })
 
-  it('Should update a playlist', async function () {
-    this.timeout(30000)
+      {
+        const res = await getAccountPlaylistsList(servers[1].url, 'root', 1, 2, 'createdAt')
 
-    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
-    })
+        expect(res.body.total).to.equal(2)
 
-    await waitJobs(servers)
+        const data: VideoPlaylist[] = res.body.data
+        expect(data).to.have.lengthOf(1)
+        expect(data[0].displayName).to.equal('playlist 3')
+      }
 
-    for (const server of servers) {
-      const res = await getVideoPlaylist(server.url, playlistServer2UUID2)
-      const playlist: VideoPlaylist = res.body
+      {
+        const res = await getAccountPlaylistsList(servers[1].url, 'root', 0, 10, 'createdAt', '3')
 
-      expect(playlist.displayName).to.equal('playlist 3 updated')
-      expect(playlist.description).to.equal('description updated')
+        expect(res.body.total).to.equal(1)
 
-      expect(playlist.privacy.id).to.equal(VideoPlaylistPrivacy.UNLISTED)
-      expect(playlist.privacy.label).to.equal('Unlisted')
+        const data: VideoPlaylist[] = res.body.data
+        expect(data).to.have.lengthOf(1)
+        expect(data[0].displayName).to.equal('playlist 3')
+      }
 
-      expect(playlist.type.id).to.equal(VideoPlaylistType.REGULAR)
-      expect(playlist.type.label).to.equal('Regular')
+      {
+        const res = await getAccountPlaylistsList(servers[1].url, 'root', 0, 10, 'createdAt', '4')
 
-      expect(playlist.videosLength).to.equal(2)
+        expect(res.body.total).to.equal(0)
 
-      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 data: VideoPlaylist[] = res.body.data
+        expect(data).to.have.lengthOf(0)
+      }
+    })
   })
 
-  it('Should create a playlist containing different startTimestamp/endTimestamp videos', async function () {
-    this.timeout(30000)
+  describe('Playlist rights', function () {
+    let unlistedPlaylist: VideoPlaylistCreateResult
+    let privatePlaylist: VideoPlaylistCreateResult
+
+    before(async function () {
+      this.timeout(30000)
+
+      {
+        const res = await createVideoPlaylist({
+          url: servers[1].url,
+          token: servers[1].accessToken,
+          playlistAttrs: {
+            displayName: 'playlist unlisted',
+            privacy: VideoPlaylistPrivacy.UNLISTED,
+            videoChannelId: servers[1].videoChannel.id
+          }
+        })
+        unlistedPlaylist = res.body.videoPlaylist
+      }
 
-    const addVideo = (elementAttrs: any) => {
-      return addVideoInPlaylist({ url: servers[0].url, token: servers[0].accessToken, playlistId: playlistServer1Id, elementAttrs })
-    }
+      {
+        const res = await createVideoPlaylist({
+          url: servers[1].url,
+          token: servers[1].accessToken,
+          playlistAttrs: {
+            displayName: 'playlist private',
+            privacy: VideoPlaylistPrivacy.PRIVATE
+          }
+        })
+        privatePlaylist = res.body.videoPlaylist
+      }
+
+      await waitJobs(servers)
+      await wait(3000)
+    })
 
-    const res = await createVideoPlaylist({
-      url: servers[ 0 ].url,
-      token: servers[ 0 ].accessToken,
-      playlistAttrs: {
-        displayName: 'playlist 4',
-        privacy: VideoPlaylistPrivacy.PUBLIC,
-        videoChannelId: servers[0].videoChannel.id
+    it('Should not list unlisted or private playlists', async function () {
+      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')
+        }
       }
     })
 
-    playlistServer1Id = res.body.videoPlaylist.id
-    playlistServer1UUID = res.body.videoPlaylist.uuid
+    it('Should not get unlisted playlist using only the id', async function () {
+      await getVideoPlaylist(servers[1].url, unlistedPlaylist.id, 404)
+    })
 
-    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 get unlisted plyaylist using uuid or shortUUID', async function () {
+      await getVideoPlaylist(servers[1].url, unlistedPlaylist.uuid)
+      await getVideoPlaylist(servers[1].url, unlistedPlaylist.shortUUID)
+    })
 
-    await waitJobs(servers)
+    it('Should not get private playlist without token', async function () {
+      for (const id of [ privatePlaylist.id, privatePlaylist.uuid, privatePlaylist.shortUUID ]) {
+        await getVideoPlaylist(servers[1].url, id, 401)
+      }
+    })
+
+    it('Should get private playlist with a token', async function () {
+      for (const id of [ privatePlaylist.id, privatePlaylist.uuid, privatePlaylist.shortUUID ]) {
+        await getVideoPlaylistWithToken(servers[1].url, servers[1].accessToken, id)
+      }
+    })
   })
 
-  it('Should correctly list playlist videos', async function () {
-    this.timeout(30000)
+  describe('Update playlists', function () {
 
-    for (const server of servers) {
-      const res = await getPlaylistVideos(server.url, server.accessToken, playlistServer1UUID, 0, 10)
+    it('Should update a playlist', async function () {
+      this.timeout(30000)
 
-      expect(res.body.total).to.equal(6)
+      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 videos: Video[] = res.body.data
-      expect(videos).to.have.lengthOf(6)
+      await waitJobs(servers)
 
-      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)
+      for (const server of servers) {
+        const res = await getVideoPlaylist(server.url, playlistServer2UUID2)
+        const playlist: VideoPlaylist = res.body
 
-      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
+        expect(playlist.displayName).to.equal('playlist 3 updated')
+        expect(playlist.description).to.equal('description updated')
 
-      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
+        expect(playlist.privacy.id).to.equal(VideoPlaylistPrivacy.UNLISTED)
+        expect(playlist.privacy.label).to.equal('Unlisted')
 
-      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)
+        expect(playlist.type.id).to.equal(VideoPlaylistType.REGULAR)
+        expect(playlist.type.label).to.equal('Regular')
 
-      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)
+        expect(playlist.videosLength).to.equal(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
+        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 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
+  describe('Element timestamps', function () {
 
-      const res3 = await getPlaylistVideos(server.url, server.accessToken, playlistServer1UUID, 0, 2)
-      expect(res3.body.data).to.have.lengthOf(2)
-    }
-  })
+    it('Should create a playlist containing different startTimestamp/endTimestamp videos', async function () {
+      this.timeout(30000)
 
-  it('Should reorder the playlist', async function () {
-    this.timeout(30000)
+      const addVideo = (elementAttrs: any) => {
+        return addVideoInPlaylist({ url: servers[0].url, token: servers[0].accessToken, playlistId: playlistServer1Id, elementAttrs })
+      }
 
-    {
-      await reorderVideosPlaylist({
-        url: servers[ 0 ].url,
-        token: servers[ 0 ].accessToken,
-        playlistId: playlistServer1Id,
-        elementAttrs: {
-          startPosition: 2,
-          insertAfterPosition: 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
         }
       })
 
-      await waitJobs(servers)
+      playlistServer1Id = res.body.videoPlaylist.id
+      playlistServer1UUID = res.body.videoPlaylist.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)
+      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
+      }
 
-        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 res = await addVideo({ videoId: servers[0].videos[4].uuid, startTimestamp: 45, stopTimestamp: 60 })
+        playlistElementServer1Video5 = res.body.videoPlaylistElement.id
       }
-    }
 
-    {
-      await reorderVideosPlaylist({
-        url: servers[0].url,
-        token: servers[0].accessToken,
-        playlistId: playlistServer1Id,
-        elementAttrs: {
-          startPosition: 1,
-          reorderLength: 3,
-          insertAfterPosition: 4
-        }
-      })
+      {
+        const res = await addVideo({ videoId: nsfwVideoServer1, startTimestamp: 5 })
+        playlistElementNSFW = res.body.videoPlaylistElement.id
+
+        await addVideo({ videoId: nsfwVideoServer1, startTimestamp: 4 })
+        await addVideo({ videoId: nsfwVideoServer1 })
+      }
 
       await waitJobs(servers)
+    })
+
+    it('Should correctly list playlist videos', async function () {
+      this.timeout(30000)
 
       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'
-        ])
+        expect(res.body.total).to.equal(8)
+
+        const videoElements: VideoPlaylistElement[] = res.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
+
+        const res3 = await getPlaylistVideos(server.url, server.accessToken, playlistServer1UUID, 0, 2)
+        expect(res3.body.data).to.have.lengthOf(2)
       }
-    }
+    })
+  })
 
-    {
-      await reorderVideosPlaylist({
+  describe('Element type', function () {
+    let groupUser1: ServerInfo[]
+    let groupWithoutToken1: ServerInfo[]
+    let group1: ServerInfo[]
+    let group2: ServerInfo[]
+
+    let video1: string
+    let video2: string
+    let video3: string
+
+    before(async function () {
+      this.timeout(60000)
+
+      groupUser1 = [ Object.assign({}, servers[0], { accessToken: userAccessTokenServer1 }) ]
+      groupWithoutToken1 = [ Object.assign({}, servers[0], { accessToken: undefined }) ]
+      group1 = [ servers[0] ]
+      group2 = [ servers[1], servers[2] ]
+
+      const res = await createVideoPlaylist({
         url: servers[0].url,
-        token: servers[0].accessToken,
-        playlistId: playlistServer1Id,
-        elementAttrs: {
-          startPosition: 6,
-          insertAfterPosition: 3
+        token: userAccessTokenServer1,
+        playlistAttrs: {
+          displayName: 'playlist 56',
+          privacy: VideoPlaylistPrivacy.PUBLIC,
+          videoChannelId: servers[0].videoChannel.id
         }
       })
 
+      const playlistServer1Id2 = res.body.videoPlaylist.id
+      playlistServer1UUID2 = res.body.videoPlaylist.uuid
+
+      const addVideo = (elementAttrs: any) => {
+        return addVideoInPlaylist({ url: servers[0].url, token: userAccessTokenServer1, playlistId: playlistServer1Id2, elementAttrs })
+      }
+
+      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
+
       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
+      await addVideo({ videoId: video1, startTimestamp: 15, stopTimestamp: 28 })
+      await addVideo({ videoId: video2, startTimestamp: 35 })
+      await addVideo({ videoId: video3 })
 
-        const names = videos.map(v => v.name)
+      await waitJobs(servers)
+    })
 
-        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'
-        ])
+    it('Should update the element type if the video is private', async function () {
+      this.timeout(20000)
 
-        for (let i = 1; i <= videos.length; i++) {
-          expect(videos[i - 1].playlistElement.position).to.equal(i)
-        }
+      const name = 'video 89'
+      const position = 1
+
+      {
+        await updateVideo(servers[0].url, servers[0].accessToken, video1, { 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 updateVideo(servers[0].url, servers[0].accessToken, video1, { 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 addVideoToBlacklist(servers[0].url, servers[0].accessToken, video1, 'reason', 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 removeVideoFromBlacklist(servers[0].url, servers[0].accessToken, 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
+      const command = servers[0].blocklistCommand
 
-      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)
+      const name = 'video 90'
+      const position = 2
 
-      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
-    }
-  })
+      {
+        await command.addToMyBlocklist({ token: userAccessTokenServer1, account: 'root@localhost:' + servers[1].port })
+        await waitJobs(servers)
 
-  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
+        await checkPlaylistElementType(groupUser1, playlistServer1UUID2, VideoPlaylistElementType.UNAVAILABLE, position, name, 3)
+        await checkPlaylistElementType(group2, playlistServer1UUID2, VideoPlaylistElementType.REGULAR, position, name, 3)
 
-    {
-      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.removeFromMyBlocklist({ token: userAccessTokenServer1, 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(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.addToMyBlocklist({ token: userAccessTokenServer1, server: 'localhost:' + servers[1].port })
+        await waitJobs(servers)
 
-    expect(obj[42000]).to.have.lengthOf(0)
-    expect(obj[43000]).to.have.lengthOf(0)
-  })
+        await checkPlaylistElementType(groupUser1, playlistServer1UUID2, VideoPlaylistElementType.UNAVAILABLE, position, name, 3)
+        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.removeFromMyBlocklist({ token: userAccessTokenServer1, 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(group2, playlistServer1UUID2, VideoPlaylistElementType.REGULAR, position, name, 3)
+      }
 
-      return (res.body.data as VideoPlaylist[]).map(p => p.displayName)
-    }
+      {
+        await command.addToServerBlocklist({ account: 'root@localhost:' + servers[1].port })
+        await waitJobs(servers)
+
+        await checkPlaylistElementType(groupUser1, playlistServer1UUID2, VideoPlaylistElementType.UNAVAILABLE, position, name, 3)
+        await checkPlaylistElementType(group2, playlistServer1UUID2, VideoPlaylistElementType.REGULAR, position, name, 3)
+
+        await command.removeFromServerBlocklist({ account: 'root@localhost:' + servers[1].port })
+        await waitJobs(servers)
+
+        await checkPlaylistElementType(group2, playlistServer1UUID2, VideoPlaylistElementType.REGULAR, position, name, 3)
+      }
 
-    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 command.addToServerBlocklist({ server: 'localhost:' + servers[1].port })
+        await waitJobs(servers)
 
-    const names1 = await getPlaylistNames()
-    expect(names1[0]).to.equal('playlist 3 updated')
-    expect(names1[1]).to.equal('playlist 2')
+        await checkPlaylistElementType(groupUser1, playlistServer1UUID2, VideoPlaylistElementType.UNAVAILABLE, position, name, 3)
+        await checkPlaylistElementType(group2, playlistServer1UUID2, VideoPlaylistElementType.REGULAR, position, name, 3)
 
-    await removeVideoFromPlaylist({ url: server.url, token: server.accessToken, playlistId: playlistServer2Id1, videoId })
+        await command.removeFromServerBlocklist({ server: 'localhost:' + servers[1].port })
+        await waitJobs(servers)
 
-    const names2 = await getPlaylistNames()
-    expect(names2[0]).to.equal('playlist 2')
-    expect(names2[1]).to.equal('playlist 3 updated')
+        await checkPlaylistElementType(group2, playlistServer1UUID2, VideoPlaylistElementType.REGULAR, position, name, 3)
+      }
+    })
+
+    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)
+
+      const elements: VideoPlaylistElement[] = res.body.data
+      const element = elements.find(e => e.position === 3)
 
-    await removeVideoFromPlaylist({ url: server.url, token: server.accessToken, playlistId: playlistServer2Id2, videoId })
+      expect(element).to.exist
+      expect(element.video).to.be.null
+      expect(element.type).to.equal(VideoPlaylistElementType.UNAVAILABLE)
+    })
 
-    const names3 = await getPlaylistNames()
-    expect(names3[0]).to.equal('playlist 3 updated')
-    expect(names3[1]).to.equal('playlist 2')
   })
 
-  it('Should delete some elements', async function () {
-    this.timeout(30000)
+  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',
+            'NSFW video',
+            'NSFW video'
+          ])
+        }
+      }
 
-    await removeVideoFromPlaylist({
-      url: servers[0].url,
-      token: servers[0].accessToken,
-      playlistId: playlistServer1Id,
-      videoId: servers[0].videos[3].uuid
+      {
+        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',
+            'NSFW video',
+            'NSFW video'
+          ])
+        }
+      }
+
+      {
+        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',
+            'NSFW video',
+            'NSFW video'
+          ])
+
+          for (let i = 1; i <= elements.length; i++) {
+            expect(elements[i - 1].position).to.equal(i)
+          }
+        }
+      }
     })
 
-    await removeVideoFromPlaylist({
-      url: servers[0].url,
-      token: servers[0].accessToken,
-      playlistId: playlistServer1Id,
-      videoId: nsfwVideoServer1
+    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
+        }
+      })
+
+      await updateVideoPlaylistElement({
+        url: servers[0].url,
+        token: servers[0].accessToken,
+        playlistId: playlistServer1Id,
+        playlistElementId: playlistElementServer1Video5,
+        elementAttrs: {
+          stopTimestamp: null
+        }
+      })
+
+      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
+
+        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)
+
+        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
+      }
     })
 
-    await waitJobs(servers)
+    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)
+      }
 
-    for (const server of servers) {
-      const res = await getPlaylistVideos(server.url, server.accessToken, playlistServer1UUID, 0, 10)
+      {
+        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(res.body.total).to.equal(4)
+      {
+        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 videos: Video[] = res.body.data
-      expect(videos).to.have.lengthOf(4)
+      expect(obj[42000]).to.have.lengthOf(0)
+      expect(obj[43000]).to.have.lengthOf(0)
+    })
 
-      expect(videos[ 0 ].name).to.equal('video 0 server 1')
-      expect(videos[ 0 ].playlistElement.position).to.equal(1)
+    it('Should automatically update updatedAt field of playlists', async function () {
+      const server = servers[1]
+      const videoId = servers[1].videos[5].id
 
-      expect(videos[ 1 ].name).to.equal('video 2 server 3')
-      expect(videos[ 1 ].playlistElement.position).to.equal(2)
+      async function getPlaylistNames () {
+        const res = await getAccountPlaylistsListWithToken(server.url, server.accessToken, 'root', 0, 5, undefined, '-updatedAt')
 
-      expect(videos[ 2 ].name).to.equal('video 1 server 3')
-      expect(videos[ 2 ].playlistElement.position).to.equal(3)
+        return (res.body.data as VideoPlaylist[]).map(p => p.displayName)
+      }
 
-      expect(videos[ 3 ].name).to.equal('video 4 server 1')
-      expect(videos[ 3 ].playlistElement.position).to.equal(4)
-    }
-  })
+      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 })
 
-  it('Should delete the playlist on server 1 and delete on server 2 and 3', async function () {
-    this.timeout(30000)
+      const element1 = res1.body.videoPlaylistElement.id
+      const element2 = res2.body.videoPlaylistElement.id
 
-    await deleteVideoPlaylist(servers[0].url, servers[0].accessToken, playlistServer1Id)
+      const names1 = await getPlaylistNames()
+      expect(names1[0]).to.equal('playlist 3 updated')
+      expect(names1[1]).to.equal('playlist 2')
 
-    await waitJobs(servers)
+      await removeVideoFromPlaylist({
+        url: server.url,
+        token: server.accessToken,
+        playlistId: playlistServer2Id1,
+        playlistElementId: element1
+      })
 
-    for (const server of servers) {
-      await getVideoPlaylist(server.url, playlistServer1UUID, 404)
-    }
-  })
+      const names2 = await getPlaylistNames()
+      expect(names2[0]).to.equal('playlist 2')
+      expect(names2[1]).to.equal('playlist 3 updated')
 
-  it('Should have deleted the thumbnail on server 1, 2 and 3', async function () {
-    this.timeout(30000)
+      await removeVideoFromPlaylist({
+        url: server.url,
+        token: server.accessToken,
+        playlistId: playlistServer2Id2,
+        playlistElementId: element2
+      })
 
-    for (const server of servers) {
-      await checkPlaylistFilesWereRemoved(playlistServer1UUID, server.serverNumber)
-    }
-  })
+      const names3 = await getPlaylistNames()
+      expect(names3[0]).to.equal('playlist 3 updated')
+      expect(names3[1]).to.equal('playlist 2')
+    })
 
-  it('Should unfollow servers 1 and 2 and hide their playlists', async function () {
-    this.timeout(30000)
+    it('Should delete some elements', async function () {
+      this.timeout(30000)
 
-    const finder = data => data.find(p => p.displayName === 'my super playlist')
+      await removeVideoFromPlaylist({
+        url: servers[0].url,
+        token: servers[0].accessToken,
+        playlistId: playlistServer1Id,
+        playlistElementId: playlistElementServer1Video4
+      })
 
-    {
-      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 removeVideoFromPlaylist({
+        url: servers[0].url,
+        token: servers[0].accessToken,
+        playlistId: playlistServer1Id,
+        playlistElementId: 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 res = await getPlaylistVideos(server.url, server.accessToken, playlistServer1UUID, 0, 10)
 
-      expect(finder(res.body.data)).to.be.undefined
-    }
-  })
+        expect(res.body.total).to.equal(6)
+
+        const elements: VideoPlaylistElement[] = res.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)
 
-  it('Should delete a channel and put the associated playlist in private mode', async function () {
-    this.timeout(30000)
+        expect(elements[1].video.name).to.equal('video 2 server 3')
+        expect(elements[1].position).to.equal(2)
 
-    const res = await addVideoChannel(servers[0].url, servers[0].accessToken, { name: 'super_channel', displayName: 'super channel' })
-    const videoChannelId = res.body.videoChannel.id
+        expect(elements[2].video.name).to.equal('video 1 server 3')
+        expect(elements[2].position).to.equal(3)
 
-    const res2 = await createVideoPlaylist({
-      url: servers[0].url,
-      token: servers[0].accessToken,
-      playlistAttrs: {
-        displayName: 'channel playlist',
-        privacy: VideoPlaylistPrivacy.PUBLIC,
-        videoChannelId
+        expect(elements[3].video.name).to.equal('video 4 server 1')
+        expect(elements[3].position).to.equal(4)
+
+        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 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
 
-    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 getVideoPlaylist(server.url, videoPlaylistIds.uuid, HttpStatusCode.OK_200)
+      }
 
-    await getVideoPlaylist(servers[1].url, videoPlaylistUUID, 404)
-  })
+      const playlistAttrs = { privacy: VideoPlaylistPrivacy.PRIVATE }
+      await updateVideoPlaylist({ url: servers[0].url, token: servers[0].accessToken, playlistId: videoPlaylistIds.id, playlistAttrs })
 
-  it('Should delete an account and delete its playlists', async function () {
-    this.timeout(30000)
+      await waitJobs(servers)
+
+      for (const server of [ servers[1], servers[2] ]) {
+        await getVideoPlaylist(server.url, videoPlaylistIds.uuid, HttpStatusCode.NOT_FOUND_404)
+      }
+      await getVideoPlaylist(servers[0].url, videoPlaylistIds.uuid, HttpStatusCode.UNAUTHORIZED_401)
 
-    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 getVideoPlaylistWithToken(servers[0].url, servers[0].accessToken, videoPlaylistIds.uuid, HttpStatusCode.OK_200)
     })
+  })
+
+  describe('Playlist deletion', function () {
+
+    it('Should delete the playlist on server 1 and delete on server 2 and 3', async function () {
+      this.timeout(30000)
 
-    const userId = res.body.user.id
-    const userAccessToken = await userLogin(servers[0], user)
+      await deleteVideoPlaylist(servers[0].url, servers[0].accessToken, playlistServer1Id)
 
-    const resChannel = await getMyUserInformation(servers[0].url, userAccessToken)
-    const userChannel = (resChannel.body as User).videoChannels[0]
+      await waitJobs(servers)
 
-    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) {
+        await getVideoPlaylist(server.url, playlistServer1UUID, HttpStatusCode.NOT_FOUND_404)
       }
     })
 
-    await waitJobs(servers)
+    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.internalServerNumber)
+      }
+    })
 
-    const finder = data => data.find(p => p.displayName === 'playlist to be deleted')
+    it('Should unfollow servers 1 and 2 and hide their playlists', async function () {
+      this.timeout(30000)
 
-    {
-      for (const server of [ servers[0], servers[1] ]) {
-        const res = await getVideoPlaylistsList(server.url, 0, 15)
+      const finder = data => data.find(p => p.displayName === 'my super playlist')
+
+      {
+        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
       }
-    }
 
-    await removeUser(servers[0].url, userId, servers[0].accessToken)
-    await waitJobs(servers)
+      await servers[2].followsCommand.unfollow({ target: servers[0] })
+
+      {
+        const res = await getVideoPlaylistsList(servers[2].url, 0, 5)
+        expect(res.body.total).to.equal(1)
 
-    {
-      for (const server of [ servers[0], servers[1] ]) {
-        const res = await getVideoPlaylistsList(server.url, 0, 15)
         expect(finder(res.body.data)).to.be.undefined
       }
-    }
+    })
+
+    it('Should delete a channel and put the associated playlist in private mode', async function () {
+      this.timeout(30000)
+
+      const res = await addVideoChannel(servers[0].url, servers[0].accessToken, { name: 'super_channel', displayName: 'super channel' })
+      const videoChannelId = res.body.videoChannel.id
+
+      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)
+
+      await deleteVideoChannel(servers[0].url, servers[0].accessToken, 'super_channel')
+
+      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 getVideoPlaylist(servers[1].url, videoPlaylistUUID, HttpStatusCode.NOT_FOUND_404)
+    })
+
+    it('Should delete an account and delete its playlists', async function () {
+      this.timeout(30000)
+
+      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
+      })
+
+      const userId = res.body.user.id
+      const userAccessToken = await userLogin(servers[0], user)
+
+      const resChannel = await getMyUserInformation(servers[0].url, userAccessToken)
+      const userChannel = (resChannel.body as User).videoChannels[0]
+
+      await createVideoPlaylist({
+        url: servers[0].url,
+        token: userAccessToken,
+        playlistAttrs: {
+          displayName: 'playlist to be deleted',
+          privacy: VideoPlaylistPrivacy.PUBLIC,
+          videoChannelId: userChannel.id
+        }
+      })
+
+      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)
+
+      {
+        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 () {
-    killallServers(servers)
-
-    // Keep the logs if the test failed
-    if (this['ok']) {
-      await flushTests()
-    }
+    await cleanupTests(servers)
   })
 })