Add playlist check param tests
This commit is contained in:
parent
418d092afa
commit
07b1a18aa6
|
@ -4,7 +4,7 @@ import {
|
||||||
asyncMiddleware,
|
asyncMiddleware,
|
||||||
asyncRetryTransactionMiddleware,
|
asyncRetryTransactionMiddleware,
|
||||||
authenticate,
|
authenticate,
|
||||||
commonVideosFiltersValidator,
|
commonVideosFiltersValidator, optionalAuthenticate,
|
||||||
paginationValidator,
|
paginationValidator,
|
||||||
setDefaultPagination,
|
setDefaultPagination,
|
||||||
setDefaultSort
|
setDefaultSort
|
||||||
|
@ -31,12 +31,14 @@ import { processImage } from '../../helpers/image-utils'
|
||||||
import { join } from 'path'
|
import { join } from 'path'
|
||||||
import { UserModel } from '../../models/account/user'
|
import { UserModel } from '../../models/account/user'
|
||||||
import {
|
import {
|
||||||
getVideoPlaylistActivityPubUrl,
|
|
||||||
getVideoPlaylistElementActivityPubUrl,
|
|
||||||
sendCreateVideoPlaylist,
|
sendCreateVideoPlaylist,
|
||||||
sendDeleteVideoPlaylist,
|
sendDeleteVideoPlaylist,
|
||||||
sendUpdateVideoPlaylist
|
sendUpdateVideoPlaylist
|
||||||
} from '../../lib/activitypub'
|
} from '../../lib/activitypub/send'
|
||||||
|
import {
|
||||||
|
getVideoPlaylistActivityPubUrl,
|
||||||
|
getVideoPlaylistElementActivityPubUrl
|
||||||
|
} from '../../lib/activitypub/url'
|
||||||
import { VideoPlaylistUpdate } from '../../../shared/models/videos/playlist/video-playlist-update.model'
|
import { VideoPlaylistUpdate } from '../../../shared/models/videos/playlist/video-playlist-update.model'
|
||||||
import { VideoModel } from '../../models/video/video'
|
import { VideoModel } from '../../models/video/video'
|
||||||
import { VideoPlaylistElementModel } from '../../models/video/video-playlist-element'
|
import { VideoPlaylistElementModel } from '../../models/video/video-playlist-element'
|
||||||
|
@ -85,6 +87,7 @@ videoPlaylistRouter.get('/:playlistId/videos',
|
||||||
asyncMiddleware(videoPlaylistsGetValidator),
|
asyncMiddleware(videoPlaylistsGetValidator),
|
||||||
paginationValidator,
|
paginationValidator,
|
||||||
setDefaultPagination,
|
setDefaultPagination,
|
||||||
|
optionalAuthenticate,
|
||||||
commonVideosFiltersValidator,
|
commonVideosFiltersValidator,
|
||||||
asyncMiddleware(getVideoPlaylistVideos)
|
asyncMiddleware(getVideoPlaylistVideos)
|
||||||
)
|
)
|
||||||
|
@ -95,7 +98,7 @@ videoPlaylistRouter.post('/:playlistId/videos',
|
||||||
asyncRetryTransactionMiddleware(addVideoInPlaylist)
|
asyncRetryTransactionMiddleware(addVideoInPlaylist)
|
||||||
)
|
)
|
||||||
|
|
||||||
videoPlaylistRouter.put('/:playlistId/videos',
|
videoPlaylistRouter.post('/:playlistId/videos/reorder',
|
||||||
authenticate,
|
authenticate,
|
||||||
asyncMiddleware(videoPlaylistsReorderVideosValidator),
|
asyncMiddleware(videoPlaylistsReorderVideosValidator),
|
||||||
asyncRetryTransactionMiddleware(reorderVideosPlaylist)
|
asyncRetryTransactionMiddleware(reorderVideosPlaylist)
|
||||||
|
@ -168,6 +171,7 @@ async function addVideoPlaylist (req: express.Request, res: express.Response) {
|
||||||
const videoPlaylistCreated: VideoPlaylistModel = await sequelizeTypescript.transaction(async t => {
|
const videoPlaylistCreated: VideoPlaylistModel = await sequelizeTypescript.transaction(async t => {
|
||||||
const videoPlaylistCreated = await videoPlaylist.save({ transaction: t })
|
const videoPlaylistCreated = await videoPlaylist.save({ transaction: t })
|
||||||
|
|
||||||
|
videoPlaylistCreated.OwnerAccount = user.Account
|
||||||
await sendCreateVideoPlaylist(videoPlaylistCreated, t)
|
await sendCreateVideoPlaylist(videoPlaylistCreated, t)
|
||||||
|
|
||||||
return videoPlaylistCreated
|
return videoPlaylistCreated
|
||||||
|
@ -349,7 +353,7 @@ async function reorderVideosPlaylist (req: express.Request, res: express.Respons
|
||||||
const videoPlaylist: VideoPlaylistModel = res.locals.videoPlaylist
|
const videoPlaylist: VideoPlaylistModel = res.locals.videoPlaylist
|
||||||
|
|
||||||
const start: number = req.body.startPosition
|
const start: number = req.body.startPosition
|
||||||
const insertAfter: number = req.body.insertAfter
|
const insertAfter: number = req.body.insertAfterPosition
|
||||||
const reorderLength: number = req.body.reorderLength || 1
|
const reorderLength: number = req.body.reorderLength || 1
|
||||||
|
|
||||||
if (start === insertAfter) {
|
if (start === insertAfter) {
|
||||||
|
|
|
@ -26,12 +26,12 @@ async function isLocalVideoChannelNameExist (name: string, res: express.Response
|
||||||
return processVideoChannelExist(videoChannel, res)
|
return processVideoChannelExist(videoChannel, res)
|
||||||
}
|
}
|
||||||
|
|
||||||
async function isVideoChannelIdExist (id: string, res: express.Response) {
|
async function isVideoChannelIdExist (id: number | string, res: express.Response) {
|
||||||
let videoChannel: VideoChannelModel
|
let videoChannel: VideoChannelModel
|
||||||
if (validator.isInt(id)) {
|
if (validator.isInt('' + id)) {
|
||||||
videoChannel = await VideoChannelModel.loadAndPopulateAccount(+id)
|
videoChannel = await VideoChannelModel.loadAndPopulateAccount(+id)
|
||||||
} else { // UUID
|
} else { // UUID
|
||||||
videoChannel = await VideoChannelModel.loadByUUIDAndPopulateAccount(id)
|
videoChannel = await VideoChannelModel.loadByUUIDAndPopulateAccount('' + id)
|
||||||
}
|
}
|
||||||
|
|
||||||
return processVideoChannelExist(videoChannel, res)
|
return processVideoChannelExist(videoChannel, res)
|
||||||
|
|
|
@ -2,6 +2,7 @@ export * from './process'
|
||||||
export * from './send'
|
export * from './send'
|
||||||
export * from './actor'
|
export * from './actor'
|
||||||
export * from './share'
|
export * from './share'
|
||||||
|
export * from './playlist'
|
||||||
export * from './videos'
|
export * from './videos'
|
||||||
export * from './video-comments'
|
export * from './video-comments'
|
||||||
export * from './video-rates'
|
export * from './video-rates'
|
||||||
|
|
|
@ -5,10 +5,8 @@ import { VideoModel } from '../../models/video/video'
|
||||||
import { VideoAbuseModel } from '../../models/video/video-abuse'
|
import { VideoAbuseModel } from '../../models/video/video-abuse'
|
||||||
import { VideoCommentModel } from '../../models/video/video-comment'
|
import { VideoCommentModel } from '../../models/video/video-comment'
|
||||||
import { VideoFileModel } from '../../models/video/video-file'
|
import { VideoFileModel } from '../../models/video/video-file'
|
||||||
import { VideoStreamingPlaylist } from '../../../shared/models/videos/video-streaming-playlist.model'
|
|
||||||
import { VideoStreamingPlaylistModel } from '../../models/video/video-streaming-playlist'
|
import { VideoStreamingPlaylistModel } from '../../models/video/video-streaming-playlist'
|
||||||
import { VideoPlaylistModel } from '../../models/video/video-playlist'
|
import { VideoPlaylistModel } from '../../models/video/video-playlist'
|
||||||
import { VideoPlaylistElementModel } from '../../models/video/video-playlist-element'
|
|
||||||
|
|
||||||
function getVideoActivityPubUrl (video: VideoModel) {
|
function getVideoActivityPubUrl (video: VideoModel) {
|
||||||
return CONFIG.WEBSERVER.URL + '/videos/watch/' + video.uuid
|
return CONFIG.WEBSERVER.URL + '/videos/watch/' + video.uuid
|
||||||
|
|
|
@ -6,7 +6,7 @@ import { UserModel } from '../../../models/account/user'
|
||||||
import { areValidationErrors } from '../utils'
|
import { areValidationErrors } from '../utils'
|
||||||
import { isVideoExist, isVideoImage } from '../../../helpers/custom-validators/videos'
|
import { isVideoExist, isVideoImage } from '../../../helpers/custom-validators/videos'
|
||||||
import { CONSTRAINTS_FIELDS } from '../../../initializers'
|
import { CONSTRAINTS_FIELDS } from '../../../initializers'
|
||||||
import { isIdOrUUIDValid, toValueOrNull } from '../../../helpers/custom-validators/misc'
|
import { isIdOrUUIDValid, isUUIDValid, toValueOrNull } from '../../../helpers/custom-validators/misc'
|
||||||
import {
|
import {
|
||||||
isVideoPlaylistDescriptionValid,
|
isVideoPlaylistDescriptionValid,
|
||||||
isVideoPlaylistExist,
|
isVideoPlaylistExist,
|
||||||
|
@ -43,10 +43,19 @@ const videoPlaylistsUpdateValidator = getCommonPlaylistEditAttributes().concat([
|
||||||
if (areValidationErrors(req, res)) return cleanUpReqFiles(req)
|
if (areValidationErrors(req, res)) return cleanUpReqFiles(req)
|
||||||
|
|
||||||
if (!await isVideoPlaylistExist(req.params.playlistId, res)) return cleanUpReqFiles(req)
|
if (!await isVideoPlaylistExist(req.params.playlistId, res)) return cleanUpReqFiles(req)
|
||||||
|
|
||||||
|
const videoPlaylist = res.locals.videoPlaylist
|
||||||
|
|
||||||
if (!checkUserCanManageVideoPlaylist(res.locals.oauth.token.User, res.locals.videoPlaylist, UserRight.REMOVE_ANY_VIDEO_PLAYLIST, res)) {
|
if (!checkUserCanManageVideoPlaylist(res.locals.oauth.token.User, res.locals.videoPlaylist, UserRight.REMOVE_ANY_VIDEO_PLAYLIST, res)) {
|
||||||
return cleanUpReqFiles(req)
|
return cleanUpReqFiles(req)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (videoPlaylist.privacy !== VideoPlaylistPrivacy.PRIVATE && req.body.privacy === VideoPlaylistPrivacy.PRIVATE) {
|
||||||
|
cleanUpReqFiles(req)
|
||||||
|
return res.status(409)
|
||||||
|
.json({ error: 'Cannot set "private" a video playlist that was not private.' })
|
||||||
|
}
|
||||||
|
|
||||||
if (req.body.videoChannelId && !await isVideoChannelIdExist(req.body.videoChannelId, res)) return cleanUpReqFiles(req)
|
if (req.body.videoChannelId && !await isVideoChannelIdExist(req.body.videoChannelId, res)) return cleanUpReqFiles(req)
|
||||||
|
|
||||||
return next()
|
return next()
|
||||||
|
@ -83,6 +92,14 @@ const videoPlaylistsGetValidator = [
|
||||||
if (!await isVideoPlaylistExist(req.params.playlistId, res)) return
|
if (!await isVideoPlaylistExist(req.params.playlistId, res)) return
|
||||||
|
|
||||||
const videoPlaylist: VideoPlaylistModel = res.locals.videoPlaylist
|
const videoPlaylist: VideoPlaylistModel = res.locals.videoPlaylist
|
||||||
|
|
||||||
|
// Video is unlisted, check we used the uuid to fetch it
|
||||||
|
if (videoPlaylist.privacy === VideoPlaylistPrivacy.UNLISTED) {
|
||||||
|
if (isUUIDValid(req.params.playlistId)) return next()
|
||||||
|
|
||||||
|
return res.status(404).end()
|
||||||
|
}
|
||||||
|
|
||||||
if (videoPlaylist.privacy === VideoPlaylistPrivacy.PRIVATE) {
|
if (videoPlaylist.privacy === VideoPlaylistPrivacy.PRIVATE) {
|
||||||
await authenticatePromiseIfNeeded(req, res)
|
await authenticatePromiseIfNeeded(req, res)
|
||||||
|
|
||||||
|
@ -121,7 +138,7 @@ const videoPlaylistsAddVideoValidator = [
|
||||||
if (areValidationErrors(req, res)) return
|
if (areValidationErrors(req, res)) return
|
||||||
|
|
||||||
if (!await isVideoPlaylistExist(req.params.playlistId, res)) return
|
if (!await isVideoPlaylistExist(req.params.playlistId, res)) return
|
||||||
if (!await isVideoExist(req.body.videoId, res, 'id')) return
|
if (!await isVideoExist(req.body.videoId, res, 'only-video')) return
|
||||||
|
|
||||||
const videoPlaylist: VideoPlaylistModel = res.locals.videoPlaylist
|
const videoPlaylist: VideoPlaylistModel = res.locals.videoPlaylist
|
||||||
const video: VideoModel = res.locals.video
|
const video: VideoModel = res.locals.video
|
||||||
|
@ -161,7 +178,7 @@ const videoPlaylistsUpdateOrRemoveVideoValidator = [
|
||||||
if (areValidationErrors(req, res)) return
|
if (areValidationErrors(req, res)) return
|
||||||
|
|
||||||
if (!await isVideoPlaylistExist(req.params.playlistId, res)) return
|
if (!await isVideoPlaylistExist(req.params.playlistId, res)) return
|
||||||
if (!await isVideoExist(req.params.playlistId, res, 'id')) return
|
if (!await isVideoExist(req.params.videoId, res, 'id')) return
|
||||||
|
|
||||||
const videoPlaylist: VideoPlaylistModel = res.locals.videoPlaylist
|
const videoPlaylist: VideoPlaylistModel = res.locals.videoPlaylist
|
||||||
const video: VideoModel = res.locals.video
|
const video: VideoModel = res.locals.video
|
||||||
|
@ -233,6 +250,27 @@ const videoPlaylistsReorderVideosValidator = [
|
||||||
const videoPlaylist: VideoPlaylistModel = res.locals.videoPlaylist
|
const videoPlaylist: VideoPlaylistModel = res.locals.videoPlaylist
|
||||||
if (!checkUserCanManageVideoPlaylist(res.locals.oauth.token.User, videoPlaylist, UserRight.UPDATE_ANY_VIDEO_PLAYLIST, res)) return
|
if (!checkUserCanManageVideoPlaylist(res.locals.oauth.token.User, videoPlaylist, UserRight.UPDATE_ANY_VIDEO_PLAYLIST, res)) return
|
||||||
|
|
||||||
|
const nextPosition = await VideoPlaylistElementModel.getNextPositionOf(videoPlaylist.id)
|
||||||
|
const startPosition: number = req.body.startPosition
|
||||||
|
const insertAfterPosition: number = req.body.insertAfterPosition
|
||||||
|
const reorderLength: number = req.body.reorderLength
|
||||||
|
|
||||||
|
if (startPosition >= nextPosition || insertAfterPosition >= nextPosition) {
|
||||||
|
res.status(400)
|
||||||
|
.json({ error: `Start position or insert after position exceed the playlist limits (max: ${nextPosition - 1})` })
|
||||||
|
.end()
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if (reorderLength && reorderLength + startPosition > nextPosition) {
|
||||||
|
res.status(400)
|
||||||
|
.json({ error: `Reorder length with this start position exceeds the playlist limits (max: ${nextPosition - startPosition})` })
|
||||||
|
.end()
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
return next()
|
return next()
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
|
|
|
@ -223,7 +223,7 @@ export class VideoChannelModel extends Model<VideoChannelModel> {
|
||||||
|
|
||||||
@HasMany(() => VideoPlaylistModel, {
|
@HasMany(() => VideoPlaylistModel, {
|
||||||
foreignKey: {
|
foreignKey: {
|
||||||
allowNull: false
|
allowNull: true
|
||||||
},
|
},
|
||||||
onDelete: 'cascade',
|
onDelete: 'cascade',
|
||||||
hooks: true
|
hooks: true
|
||||||
|
|
|
@ -188,7 +188,8 @@ export class VideoPlaylistElementModel extends Model<VideoPlaylistElementModel>
|
||||||
[Sequelize.Op.lte]: endPosition
|
[Sequelize.Op.lte]: endPosition
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
transaction
|
transaction,
|
||||||
|
validate: false // We use a literal to update the position
|
||||||
}
|
}
|
||||||
|
|
||||||
return VideoPlaylistElementModel.update({ position: Sequelize.literal(`${newPosition} + "position" - ${firstPosition}`) }, query)
|
return VideoPlaylistElementModel.update({ position: Sequelize.literal(`${newPosition} + "position" - ${firstPosition}`) }, query)
|
||||||
|
|
|
@ -197,7 +197,7 @@ export class VideoPlaylistModel extends Model<VideoPlaylistModel> {
|
||||||
|
|
||||||
@BelongsTo(() => VideoChannelModel, {
|
@BelongsTo(() => VideoChannelModel, {
|
||||||
foreignKey: {
|
foreignKey: {
|
||||||
allowNull: false
|
allowNull: true
|
||||||
},
|
},
|
||||||
onDelete: 'CASCADE'
|
onDelete: 'CASCADE'
|
||||||
})
|
})
|
||||||
|
@ -351,7 +351,7 @@ export class VideoPlaylistModel extends Model<VideoPlaylistModel> {
|
||||||
updatedAt: this.updatedAt,
|
updatedAt: this.updatedAt,
|
||||||
|
|
||||||
ownerAccount: this.OwnerAccount.toFormattedSummaryJSON(),
|
ownerAccount: this.OwnerAccount.toFormattedSummaryJSON(),
|
||||||
videoChannel: this.VideoChannel.toFormattedSummaryJSON()
|
videoChannel: this.VideoChannel ? this.VideoChannel.toFormattedSummaryJSON() : null
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,35 +1,35 @@
|
||||||
/* tslint:disable:no-unused-expression */
|
/* tslint:disable:no-unused-expression */
|
||||||
|
|
||||||
import { omit } from 'lodash'
|
|
||||||
import 'mocha'
|
import 'mocha'
|
||||||
import { join } from 'path'
|
|
||||||
import { VideoPrivacy } from '../../../../shared/models/videos/video-privacy.enum'
|
|
||||||
import {
|
import {
|
||||||
createUser,
|
createUser,
|
||||||
|
createVideoPlaylist,
|
||||||
|
deleteVideoPlaylist,
|
||||||
flushTests,
|
flushTests,
|
||||||
getMyUserInformation,
|
getVideoPlaylist,
|
||||||
immutableAssign,
|
immutableAssign,
|
||||||
killallServers,
|
killallServers,
|
||||||
makeGetRequest,
|
makeGetRequest,
|
||||||
makePostBodyRequest,
|
|
||||||
makeUploadRequest,
|
|
||||||
runServer,
|
runServer,
|
||||||
ServerInfo,
|
ServerInfo,
|
||||||
setAccessTokensToServers,
|
setAccessTokensToServers,
|
||||||
updateCustomSubConfig,
|
updateVideoPlaylist,
|
||||||
userLogin
|
userLogin,
|
||||||
|
addVideoInPlaylist, uploadVideo, updateVideoPlaylistElement, removeVideoFromPlaylist, reorderVideosPlaylist
|
||||||
} from '../../../../shared/utils'
|
} from '../../../../shared/utils'
|
||||||
import {
|
import {
|
||||||
checkBadCountPagination,
|
checkBadCountPagination,
|
||||||
checkBadSortPagination,
|
checkBadSortPagination,
|
||||||
checkBadStartPagination
|
checkBadStartPagination
|
||||||
} from '../../../../shared/utils/requests/check-api-params'
|
} from '../../../../shared/utils/requests/check-api-params'
|
||||||
import { getMagnetURI, getYoutubeVideoUrl } from '../../../../shared/utils/videos/video-imports'
|
import { VideoPlaylistPrivacy } from '../../../../shared/models/videos/playlist/video-playlist-privacy.model'
|
||||||
|
|
||||||
describe('Test video playlists API validator', function () {
|
describe('Test video playlists API validator', function () {
|
||||||
const path = '/api/v1/videos/video-playlists'
|
|
||||||
let server: ServerInfo
|
let server: ServerInfo
|
||||||
let userAccessToken = ''
|
let userAccessToken = ''
|
||||||
|
let playlistUUID: string
|
||||||
|
let videoId: number
|
||||||
|
let videoId2: number
|
||||||
|
|
||||||
// ---------------------------------------------------------------
|
// ---------------------------------------------------------------
|
||||||
|
|
||||||
|
@ -46,9 +46,31 @@ describe('Test video playlists API validator', function () {
|
||||||
const password = 'my super password'
|
const password = 'my super password'
|
||||||
await createUser(server.url, server.accessToken, username, password)
|
await createUser(server.url, server.accessToken, username, password)
|
||||||
userAccessToken = await userLogin(server, { username, password })
|
userAccessToken = await userLogin(server, { username, password })
|
||||||
|
|
||||||
|
{
|
||||||
|
const res = await uploadVideo(server.url, server.accessToken, { name: 'video 1' })
|
||||||
|
videoId = res.body.video.id
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
const res = await uploadVideo(server.url, server.accessToken, { name: 'video 2' })
|
||||||
|
videoId2 = res.body.video.id
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
const res = await createVideoPlaylist({
|
||||||
|
url: server.url,
|
||||||
|
token: server.accessToken,
|
||||||
|
playlistAttrs: {
|
||||||
|
displayName: 'super playlist',
|
||||||
|
privacy: VideoPlaylistPrivacy.PUBLIC
|
||||||
|
}
|
||||||
|
})
|
||||||
|
playlistUUID = res.body.videoPlaylist.uuid
|
||||||
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
describe('When listing video playlists', function () {
|
describe('When listing playlists', function () {
|
||||||
const globalPath = '/api/v1/video-playlists'
|
const globalPath = '/api/v1/video-playlists'
|
||||||
const accountPath = '/api/v1/accounts/root/video-playlists'
|
const accountPath = '/api/v1/accounts/root/video-playlists'
|
||||||
const videoChannelPath = '/api/v1/video-channels/root_channel/video-playlists'
|
const videoChannelPath = '/api/v1/video-channels/root_channel/video-playlists'
|
||||||
|
@ -90,7 +112,7 @@ describe('Test video playlists API validator', function () {
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
describe('When listing videos of a playlist', async function () {
|
describe('When listing videos of a playlist', function () {
|
||||||
const path = '/api/v1/video-playlists'
|
const path = '/api/v1/video-playlists'
|
||||||
|
|
||||||
it('Should fail with a bad start pagination', async function () {
|
it('Should fail with a bad start pagination', async function () {
|
||||||
|
@ -101,11 +123,743 @@ describe('Test video playlists API validator', function () {
|
||||||
await checkBadCountPagination(server.url, path, server.accessToken)
|
await checkBadCountPagination(server.url, path, server.accessToken)
|
||||||
})
|
})
|
||||||
|
|
||||||
it('Should fail with an incorrect sort', async function () {
|
it('Should fail with a bad filter', async function () {
|
||||||
await checkBadSortPagination(server.url, path, server.accessToken)
|
await checkBadSortPagination(server.url, path, server.accessToken)
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
|
describe('When getting a video playlist', function () {
|
||||||
|
it('Should fail with a bad id or uuid', async function () {
|
||||||
|
await getVideoPlaylist(server.url, 'toto', 400)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should fail with an unknown playlist', async function () {
|
||||||
|
await getVideoPlaylist(server.url, 42, 404)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should fail to get an unlisted playlist with the number id', async function () {
|
||||||
|
const res = await createVideoPlaylist({
|
||||||
|
url: server.url,
|
||||||
|
token: server.accessToken,
|
||||||
|
playlistAttrs: {
|
||||||
|
displayName: 'super playlist',
|
||||||
|
privacy: VideoPlaylistPrivacy.UNLISTED
|
||||||
|
}
|
||||||
|
})
|
||||||
|
const playlist = res.body.videoPlaylist
|
||||||
|
|
||||||
|
await getVideoPlaylist(server.url, playlist.id, 404)
|
||||||
|
await getVideoPlaylist(server.url, playlist.uuid, 200)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should succeed with the correct params', async function () {
|
||||||
|
await getVideoPlaylist(server.url, playlistUUID, 200)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('When creating/updating a video playlist', function () {
|
||||||
|
|
||||||
|
it('Should fail with an unauthenticated user', async function () {
|
||||||
|
const baseParams = {
|
||||||
|
url: server.url,
|
||||||
|
token: null,
|
||||||
|
playlistAttrs: {
|
||||||
|
displayName: 'super playlist',
|
||||||
|
privacy: VideoPlaylistPrivacy.PUBLIC
|
||||||
|
},
|
||||||
|
expectedStatus: 401
|
||||||
|
}
|
||||||
|
|
||||||
|
await createVideoPlaylist(baseParams)
|
||||||
|
await updateVideoPlaylist(immutableAssign(baseParams, { playlistId: playlistUUID }))
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should fail without displayName', async function () {
|
||||||
|
const baseParams = {
|
||||||
|
url: server.url,
|
||||||
|
token: server.accessToken,
|
||||||
|
playlistAttrs: {
|
||||||
|
privacy: VideoPlaylistPrivacy.PUBLIC
|
||||||
|
} as any,
|
||||||
|
expectedStatus: 400
|
||||||
|
}
|
||||||
|
|
||||||
|
await createVideoPlaylist(baseParams)
|
||||||
|
await updateVideoPlaylist(immutableAssign(baseParams, { playlistId: playlistUUID }))
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should fail with an incorrect display name', async function () {
|
||||||
|
const baseParams = {
|
||||||
|
url: server.url,
|
||||||
|
token: server.accessToken,
|
||||||
|
playlistAttrs: {
|
||||||
|
displayName: 's'.repeat(300),
|
||||||
|
privacy: VideoPlaylistPrivacy.PUBLIC
|
||||||
|
},
|
||||||
|
expectedStatus: 400
|
||||||
|
}
|
||||||
|
|
||||||
|
await createVideoPlaylist(baseParams)
|
||||||
|
await updateVideoPlaylist(immutableAssign(baseParams, { playlistId: playlistUUID }))
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should fail with an incorrect description', async function () {
|
||||||
|
const baseParams = {
|
||||||
|
url: server.url,
|
||||||
|
token: server.accessToken,
|
||||||
|
playlistAttrs: {
|
||||||
|
displayName: 'display name',
|
||||||
|
privacy: VideoPlaylistPrivacy.PUBLIC,
|
||||||
|
description: 't'
|
||||||
|
},
|
||||||
|
expectedStatus: 400
|
||||||
|
}
|
||||||
|
|
||||||
|
await createVideoPlaylist(baseParams)
|
||||||
|
await updateVideoPlaylist(immutableAssign(baseParams, { playlistId: playlistUUID }))
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should fail with an incorrect privacy', async function () {
|
||||||
|
const baseParams = {
|
||||||
|
url: server.url,
|
||||||
|
token: server.accessToken,
|
||||||
|
playlistAttrs: {
|
||||||
|
displayName: 'display name',
|
||||||
|
privacy: 45
|
||||||
|
} as any,
|
||||||
|
expectedStatus: 400
|
||||||
|
}
|
||||||
|
|
||||||
|
await createVideoPlaylist(baseParams)
|
||||||
|
await updateVideoPlaylist(immutableAssign(baseParams, { playlistId: playlistUUID }))
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should fail with an unknown video channel id', async function () {
|
||||||
|
const baseParams = {
|
||||||
|
url: server.url,
|
||||||
|
token: server.accessToken,
|
||||||
|
playlistAttrs: {
|
||||||
|
displayName: 'display name',
|
||||||
|
privacy: VideoPlaylistPrivacy.PUBLIC,
|
||||||
|
videoChannelId: 42
|
||||||
|
},
|
||||||
|
expectedStatus: 404
|
||||||
|
}
|
||||||
|
|
||||||
|
await createVideoPlaylist(baseParams)
|
||||||
|
await updateVideoPlaylist(immutableAssign(baseParams, { playlistId: playlistUUID }))
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should fail with an incorrect thumbnail file', async function () {
|
||||||
|
const baseParams = {
|
||||||
|
url: server.url,
|
||||||
|
token: server.accessToken,
|
||||||
|
playlistAttrs: {
|
||||||
|
displayName: 'display name',
|
||||||
|
privacy: VideoPlaylistPrivacy.PUBLIC,
|
||||||
|
thumbnailfile: 'avatar.png'
|
||||||
|
},
|
||||||
|
expectedStatus: 400
|
||||||
|
}
|
||||||
|
|
||||||
|
await createVideoPlaylist(baseParams)
|
||||||
|
await updateVideoPlaylist(immutableAssign(baseParams, { playlistId: playlistUUID }))
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should fail with an unknown playlist to update', async function () {
|
||||||
|
await updateVideoPlaylist({
|
||||||
|
url: server.url,
|
||||||
|
token: server.accessToken,
|
||||||
|
playlistId: 42,
|
||||||
|
playlistAttrs: {
|
||||||
|
displayName: 'display name',
|
||||||
|
privacy: VideoPlaylistPrivacy.PUBLIC
|
||||||
|
},
|
||||||
|
expectedStatus: 404
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should fail to update a playlist of another user', async function () {
|
||||||
|
await updateVideoPlaylist({
|
||||||
|
url: server.url,
|
||||||
|
token: userAccessToken,
|
||||||
|
playlistId: playlistUUID,
|
||||||
|
playlistAttrs: {
|
||||||
|
displayName: 'display name',
|
||||||
|
privacy: VideoPlaylistPrivacy.PUBLIC
|
||||||
|
},
|
||||||
|
expectedStatus: 403
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should fail to update to private a public/unlisted playlist', async function () {
|
||||||
|
const res = await createVideoPlaylist({
|
||||||
|
url: server.url,
|
||||||
|
token: server.accessToken,
|
||||||
|
playlistAttrs: {
|
||||||
|
displayName: 'super playlist',
|
||||||
|
privacy: VideoPlaylistPrivacy.PUBLIC
|
||||||
|
}
|
||||||
|
})
|
||||||
|
const playlist = res.body.videoPlaylist
|
||||||
|
|
||||||
|
await updateVideoPlaylist({
|
||||||
|
url: server.url,
|
||||||
|
token: server.accessToken,
|
||||||
|
playlistId: playlist.id,
|
||||||
|
playlistAttrs: {
|
||||||
|
displayName: 'display name',
|
||||||
|
privacy: VideoPlaylistPrivacy.PRIVATE
|
||||||
|
},
|
||||||
|
expectedStatus: 409
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should succeed with the correct params', async function () {
|
||||||
|
const baseParams = {
|
||||||
|
url: server.url,
|
||||||
|
token: server.accessToken,
|
||||||
|
playlistAttrs: {
|
||||||
|
displayName: 'display name',
|
||||||
|
privacy: VideoPlaylistPrivacy.UNLISTED,
|
||||||
|
thumbnailfile: 'thumbnail.jpg'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
await createVideoPlaylist(baseParams)
|
||||||
|
await updateVideoPlaylist(immutableAssign(baseParams, { playlistId: playlistUUID }))
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('When adding an element in a playlist', function () {
|
||||||
|
it('Should fail with an unauthenticated user', async function () {
|
||||||
|
await addVideoInPlaylist({
|
||||||
|
url: server.url,
|
||||||
|
token: null,
|
||||||
|
elementAttrs: {
|
||||||
|
videoId: videoId
|
||||||
|
},
|
||||||
|
playlistId: playlistUUID,
|
||||||
|
expectedStatus: 401
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should fail with the playlist of another user', async function () {
|
||||||
|
await addVideoInPlaylist({
|
||||||
|
url: server.url,
|
||||||
|
token: userAccessToken,
|
||||||
|
elementAttrs: {
|
||||||
|
videoId: videoId
|
||||||
|
},
|
||||||
|
playlistId: playlistUUID,
|
||||||
|
expectedStatus: 403
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should fail with an unknown or incorrect playlist id', async function () {
|
||||||
|
await addVideoInPlaylist({
|
||||||
|
url: server.url,
|
||||||
|
token: server.accessToken,
|
||||||
|
elementAttrs: {
|
||||||
|
videoId: videoId
|
||||||
|
},
|
||||||
|
playlistId: 'toto',
|
||||||
|
expectedStatus: 400
|
||||||
|
})
|
||||||
|
|
||||||
|
await addVideoInPlaylist({
|
||||||
|
url: server.url,
|
||||||
|
token: server.accessToken,
|
||||||
|
elementAttrs: {
|
||||||
|
videoId: videoId
|
||||||
|
},
|
||||||
|
playlistId: 42,
|
||||||
|
expectedStatus: 404
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should fail with an unknown or incorrect video id', async function () {
|
||||||
|
await addVideoInPlaylist({
|
||||||
|
url: server.url,
|
||||||
|
token: server.accessToken,
|
||||||
|
elementAttrs: {
|
||||||
|
videoId: 'toto' as any
|
||||||
|
},
|
||||||
|
playlistId: playlistUUID,
|
||||||
|
expectedStatus: 400
|
||||||
|
})
|
||||||
|
|
||||||
|
await addVideoInPlaylist({
|
||||||
|
url: server.url,
|
||||||
|
token: server.accessToken,
|
||||||
|
elementAttrs: {
|
||||||
|
videoId: 42
|
||||||
|
},
|
||||||
|
playlistId: playlistUUID,
|
||||||
|
expectedStatus: 404
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should fail with a bad start/stop timestamp', async function () {
|
||||||
|
await addVideoInPlaylist({
|
||||||
|
url: server.url,
|
||||||
|
token: server.accessToken,
|
||||||
|
elementAttrs: {
|
||||||
|
videoId: videoId,
|
||||||
|
startTimestamp: -42
|
||||||
|
},
|
||||||
|
playlistId: playlistUUID,
|
||||||
|
expectedStatus: 400
|
||||||
|
})
|
||||||
|
|
||||||
|
await addVideoInPlaylist({
|
||||||
|
url: server.url,
|
||||||
|
token: server.accessToken,
|
||||||
|
elementAttrs: {
|
||||||
|
videoId: videoId,
|
||||||
|
stopTimestamp: 'toto' as any
|
||||||
|
},
|
||||||
|
playlistId: playlistUUID,
|
||||||
|
expectedStatus: 400
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Succeed with the correct params', async function () {
|
||||||
|
await addVideoInPlaylist({
|
||||||
|
url: server.url,
|
||||||
|
token: server.accessToken,
|
||||||
|
elementAttrs: {
|
||||||
|
videoId: videoId,
|
||||||
|
stopTimestamp: 3
|
||||||
|
},
|
||||||
|
playlistId: playlistUUID,
|
||||||
|
expectedStatus: 200
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should fail if the video was already added in the playlist', async function () {
|
||||||
|
await addVideoInPlaylist({
|
||||||
|
url: server.url,
|
||||||
|
token: server.accessToken,
|
||||||
|
elementAttrs: {
|
||||||
|
videoId: videoId,
|
||||||
|
stopTimestamp: 3
|
||||||
|
},
|
||||||
|
playlistId: playlistUUID,
|
||||||
|
expectedStatus: 409
|
||||||
|
})
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('When updating an element in a playlist', function () {
|
||||||
|
it('Should fail with an unauthenticated user', async function () {
|
||||||
|
await updateVideoPlaylistElement({
|
||||||
|
url: server.url,
|
||||||
|
token: null,
|
||||||
|
elementAttrs: { },
|
||||||
|
videoId: videoId,
|
||||||
|
playlistId: playlistUUID,
|
||||||
|
expectedStatus: 401
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should fail with the playlist of another user', async function () {
|
||||||
|
await updateVideoPlaylistElement({
|
||||||
|
url: server.url,
|
||||||
|
token: userAccessToken,
|
||||||
|
elementAttrs: { },
|
||||||
|
videoId: videoId,
|
||||||
|
playlistId: playlistUUID,
|
||||||
|
expectedStatus: 403
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should fail with an unknown or incorrect playlist id', async function () {
|
||||||
|
await updateVideoPlaylistElement({
|
||||||
|
url: server.url,
|
||||||
|
token: server.accessToken,
|
||||||
|
elementAttrs: { },
|
||||||
|
videoId: videoId,
|
||||||
|
playlistId: 'toto',
|
||||||
|
expectedStatus: 400
|
||||||
|
})
|
||||||
|
|
||||||
|
await updateVideoPlaylistElement({
|
||||||
|
url: server.url,
|
||||||
|
token: server.accessToken,
|
||||||
|
elementAttrs: { },
|
||||||
|
videoId: videoId,
|
||||||
|
playlistId: 42,
|
||||||
|
expectedStatus: 404
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should fail with an unknown or incorrect video id', async function () {
|
||||||
|
await updateVideoPlaylistElement({
|
||||||
|
url: server.url,
|
||||||
|
token: server.accessToken,
|
||||||
|
elementAttrs: { },
|
||||||
|
videoId: 'toto',
|
||||||
|
playlistId: playlistUUID,
|
||||||
|
expectedStatus: 400
|
||||||
|
})
|
||||||
|
|
||||||
|
await updateVideoPlaylistElement({
|
||||||
|
url: server.url,
|
||||||
|
token: server.accessToken,
|
||||||
|
elementAttrs: { },
|
||||||
|
videoId: 42,
|
||||||
|
playlistId: playlistUUID,
|
||||||
|
expectedStatus: 404
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should fail with a bad start/stop timestamp', async function () {
|
||||||
|
await updateVideoPlaylistElement({
|
||||||
|
url: server.url,
|
||||||
|
token: server.accessToken,
|
||||||
|
elementAttrs: {
|
||||||
|
startTimestamp: 'toto' as any
|
||||||
|
},
|
||||||
|
videoId: videoId,
|
||||||
|
playlistId: playlistUUID,
|
||||||
|
expectedStatus: 400
|
||||||
|
})
|
||||||
|
|
||||||
|
await updateVideoPlaylistElement({
|
||||||
|
url: server.url,
|
||||||
|
token: server.accessToken,
|
||||||
|
elementAttrs: {
|
||||||
|
stopTimestamp: -42
|
||||||
|
},
|
||||||
|
videoId: videoId,
|
||||||
|
playlistId: playlistUUID,
|
||||||
|
expectedStatus: 400
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should fail with an unknown element', async function () {
|
||||||
|
await updateVideoPlaylistElement({
|
||||||
|
url: server.url,
|
||||||
|
token: server.accessToken,
|
||||||
|
elementAttrs: {
|
||||||
|
stopTimestamp: 2
|
||||||
|
},
|
||||||
|
videoId: videoId2,
|
||||||
|
playlistId: playlistUUID,
|
||||||
|
expectedStatus: 404
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Succeed with the correct params', async function () {
|
||||||
|
await updateVideoPlaylistElement({
|
||||||
|
url: server.url,
|
||||||
|
token: server.accessToken,
|
||||||
|
elementAttrs: {
|
||||||
|
stopTimestamp: 2
|
||||||
|
},
|
||||||
|
videoId: videoId,
|
||||||
|
playlistId: playlistUUID,
|
||||||
|
expectedStatus: 204
|
||||||
|
})
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('When reordering elements of a playlist', function () {
|
||||||
|
let videoId3: number
|
||||||
|
let videoId4: number
|
||||||
|
|
||||||
|
before(async function () {
|
||||||
|
{
|
||||||
|
const res = await uploadVideo(server.url, server.accessToken, { name: 'video 3' })
|
||||||
|
videoId3 = res.body.video.id
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
const res = await uploadVideo(server.url, server.accessToken, { name: 'video 4' })
|
||||||
|
videoId4 = res.body.video.id
|
||||||
|
}
|
||||||
|
|
||||||
|
await addVideoInPlaylist({
|
||||||
|
url: server.url,
|
||||||
|
token: server.accessToken,
|
||||||
|
playlistId: playlistUUID,
|
||||||
|
elementAttrs: { videoId: videoId3 }
|
||||||
|
})
|
||||||
|
|
||||||
|
await addVideoInPlaylist({
|
||||||
|
url: server.url,
|
||||||
|
token: server.accessToken,
|
||||||
|
playlistId: playlistUUID,
|
||||||
|
elementAttrs: { videoId: videoId4 }
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should fail with an unauthenticated user', async function () {
|
||||||
|
await reorderVideosPlaylist({
|
||||||
|
url: server.url,
|
||||||
|
token: null,
|
||||||
|
playlistId: playlistUUID,
|
||||||
|
elementAttrs: {
|
||||||
|
startPosition: 1,
|
||||||
|
insertAfterPosition: 2
|
||||||
|
},
|
||||||
|
expectedStatus: 401
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should fail with the playlist of another user', async function () {
|
||||||
|
await reorderVideosPlaylist({
|
||||||
|
url: server.url,
|
||||||
|
token: userAccessToken,
|
||||||
|
playlistId: playlistUUID,
|
||||||
|
elementAttrs: {
|
||||||
|
startPosition: 1,
|
||||||
|
insertAfterPosition: 2
|
||||||
|
},
|
||||||
|
expectedStatus: 403
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should fail with an invalid playlist', async function () {
|
||||||
|
await reorderVideosPlaylist({
|
||||||
|
url: server.url,
|
||||||
|
token: server.accessToken,
|
||||||
|
playlistId: 'toto',
|
||||||
|
elementAttrs: {
|
||||||
|
startPosition: 1,
|
||||||
|
insertAfterPosition: 2
|
||||||
|
},
|
||||||
|
expectedStatus: 400
|
||||||
|
})
|
||||||
|
|
||||||
|
await reorderVideosPlaylist({
|
||||||
|
url: server.url,
|
||||||
|
token: server.accessToken,
|
||||||
|
playlistId: 42,
|
||||||
|
elementAttrs: {
|
||||||
|
startPosition: 1,
|
||||||
|
insertAfterPosition: 2
|
||||||
|
},
|
||||||
|
expectedStatus: 404
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should fail with an invalid start position', async function () {
|
||||||
|
await reorderVideosPlaylist({
|
||||||
|
url: server.url,
|
||||||
|
token: server.accessToken,
|
||||||
|
playlistId: playlistUUID,
|
||||||
|
elementAttrs: {
|
||||||
|
startPosition: -1,
|
||||||
|
insertAfterPosition: 2
|
||||||
|
},
|
||||||
|
expectedStatus: 400
|
||||||
|
})
|
||||||
|
|
||||||
|
await reorderVideosPlaylist({
|
||||||
|
url: server.url,
|
||||||
|
token: server.accessToken,
|
||||||
|
playlistId: playlistUUID,
|
||||||
|
elementAttrs: {
|
||||||
|
startPosition: 'toto' as any,
|
||||||
|
insertAfterPosition: 2
|
||||||
|
},
|
||||||
|
expectedStatus: 400
|
||||||
|
})
|
||||||
|
|
||||||
|
await reorderVideosPlaylist({
|
||||||
|
url: server.url,
|
||||||
|
token: server.accessToken,
|
||||||
|
playlistId: playlistUUID,
|
||||||
|
elementAttrs: {
|
||||||
|
startPosition: 42,
|
||||||
|
insertAfterPosition: 2
|
||||||
|
},
|
||||||
|
expectedStatus: 400
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should fail with an invalid insert after position', async function () {
|
||||||
|
await reorderVideosPlaylist({
|
||||||
|
url: server.url,
|
||||||
|
token: server.accessToken,
|
||||||
|
playlistId: playlistUUID,
|
||||||
|
elementAttrs: {
|
||||||
|
startPosition: 1,
|
||||||
|
insertAfterPosition: 'toto' as any
|
||||||
|
},
|
||||||
|
expectedStatus: 400
|
||||||
|
})
|
||||||
|
|
||||||
|
await reorderVideosPlaylist({
|
||||||
|
url: server.url,
|
||||||
|
token: server.accessToken,
|
||||||
|
playlistId: playlistUUID,
|
||||||
|
elementAttrs: {
|
||||||
|
startPosition: 1,
|
||||||
|
insertAfterPosition: -2
|
||||||
|
},
|
||||||
|
expectedStatus: 400
|
||||||
|
})
|
||||||
|
|
||||||
|
await reorderVideosPlaylist({
|
||||||
|
url: server.url,
|
||||||
|
token: server.accessToken,
|
||||||
|
playlistId: playlistUUID,
|
||||||
|
elementAttrs: {
|
||||||
|
startPosition: 1,
|
||||||
|
insertAfterPosition: 42
|
||||||
|
},
|
||||||
|
expectedStatus: 400
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should fail with an invalid reorder length', async function () {
|
||||||
|
await reorderVideosPlaylist({
|
||||||
|
url: server.url,
|
||||||
|
token: server.accessToken,
|
||||||
|
playlistId: playlistUUID,
|
||||||
|
elementAttrs: {
|
||||||
|
startPosition: 1,
|
||||||
|
insertAfterPosition: 2,
|
||||||
|
reorderLength: 'toto' as any
|
||||||
|
},
|
||||||
|
expectedStatus: 400
|
||||||
|
})
|
||||||
|
|
||||||
|
await reorderVideosPlaylist({
|
||||||
|
url: server.url,
|
||||||
|
token: server.accessToken,
|
||||||
|
playlistId: playlistUUID,
|
||||||
|
elementAttrs: {
|
||||||
|
startPosition: 1,
|
||||||
|
insertAfterPosition: 2,
|
||||||
|
reorderLength: -1
|
||||||
|
},
|
||||||
|
expectedStatus: 400
|
||||||
|
})
|
||||||
|
|
||||||
|
await reorderVideosPlaylist({
|
||||||
|
url: server.url,
|
||||||
|
token: server.accessToken,
|
||||||
|
playlistId: playlistUUID,
|
||||||
|
elementAttrs: {
|
||||||
|
startPosition: 1,
|
||||||
|
insertAfterPosition: 2,
|
||||||
|
reorderLength: 4
|
||||||
|
},
|
||||||
|
expectedStatus: 400
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Succeed with the correct params', async function () {
|
||||||
|
await reorderVideosPlaylist({
|
||||||
|
url: server.url,
|
||||||
|
token: server.accessToken,
|
||||||
|
playlistId: playlistUUID,
|
||||||
|
elementAttrs: {
|
||||||
|
startPosition: 1,
|
||||||
|
insertAfterPosition: 2,
|
||||||
|
reorderLength: 3
|
||||||
|
},
|
||||||
|
expectedStatus: 204
|
||||||
|
})
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('When deleting an element in a playlist', function () {
|
||||||
|
it('Should fail with an unauthenticated user', async function () {
|
||||||
|
await removeVideoFromPlaylist({
|
||||||
|
url: server.url,
|
||||||
|
token: null,
|
||||||
|
videoId,
|
||||||
|
playlistId: playlistUUID,
|
||||||
|
expectedStatus: 401
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should fail with the playlist of another user', async function () {
|
||||||
|
await removeVideoFromPlaylist({
|
||||||
|
url: server.url,
|
||||||
|
token: userAccessToken,
|
||||||
|
videoId,
|
||||||
|
playlistId: playlistUUID,
|
||||||
|
expectedStatus: 403
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should fail with an unknown or incorrect playlist id', async function () {
|
||||||
|
await removeVideoFromPlaylist({
|
||||||
|
url: server.url,
|
||||||
|
token: server.accessToken,
|
||||||
|
videoId,
|
||||||
|
playlistId: 'toto',
|
||||||
|
expectedStatus: 400
|
||||||
|
})
|
||||||
|
|
||||||
|
await removeVideoFromPlaylist({
|
||||||
|
url: server.url,
|
||||||
|
token: server.accessToken,
|
||||||
|
videoId,
|
||||||
|
playlistId: 42,
|
||||||
|
expectedStatus: 404
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should fail with an unknown or incorrect video id', async function () {
|
||||||
|
await removeVideoFromPlaylist({
|
||||||
|
url: server.url,
|
||||||
|
token: server.accessToken,
|
||||||
|
videoId: 'toto',
|
||||||
|
playlistId: playlistUUID,
|
||||||
|
expectedStatus: 400
|
||||||
|
})
|
||||||
|
|
||||||
|
await removeVideoFromPlaylist({
|
||||||
|
url: server.url,
|
||||||
|
token: server.accessToken,
|
||||||
|
videoId: 42,
|
||||||
|
playlistId: playlistUUID,
|
||||||
|
expectedStatus: 404
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should fail with an unknown element', async function () {
|
||||||
|
await removeVideoFromPlaylist({
|
||||||
|
url: server.url,
|
||||||
|
token: server.accessToken,
|
||||||
|
videoId: videoId2,
|
||||||
|
playlistId: playlistUUID,
|
||||||
|
expectedStatus: 404
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Succeed with the correct params', async function () {
|
||||||
|
await removeVideoFromPlaylist({
|
||||||
|
url: server.url,
|
||||||
|
token: server.accessToken,
|
||||||
|
videoId: videoId,
|
||||||
|
playlistId: playlistUUID,
|
||||||
|
expectedStatus: 204
|
||||||
|
})
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('When deleting a playlist', function () {
|
||||||
|
it('Should fail with an unknown playlist', async function () {
|
||||||
|
await deleteVideoPlaylist(server.url, server.accessToken, 42, 404)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should fail with a playlist of another user', async function () {
|
||||||
|
await deleteVideoPlaylist(server.url, userAccessToken, playlistUUID, 403)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should succeed with the correct params', async function () {
|
||||||
|
await deleteVideoPlaylist(server.url, server.accessToken, playlistUUID)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
after(async function () {
|
after(async function () {
|
||||||
killallServers([ server ])
|
killallServers([ server ])
|
||||||
|
|
||||||
|
|
|
@ -1,9 +1,9 @@
|
||||||
/* tslint:disable:no-unused-expression */
|
/* tslint:disable:no-unused-expression */
|
||||||
|
|
||||||
import * as chai from 'chai'
|
|
||||||
import 'mocha'
|
import 'mocha'
|
||||||
import {
|
import {
|
||||||
createUser,
|
createUser,
|
||||||
|
createVideoPlaylist,
|
||||||
flushTests,
|
flushTests,
|
||||||
killallServers,
|
killallServers,
|
||||||
makeGetRequest,
|
makeGetRequest,
|
||||||
|
@ -13,15 +13,15 @@ import {
|
||||||
userLogin
|
userLogin
|
||||||
} from '../../../../shared/utils'
|
} from '../../../../shared/utils'
|
||||||
import { UserRole } from '../../../../shared/models/users'
|
import { UserRole } from '../../../../shared/models/users'
|
||||||
|
import { VideoPlaylistPrivacy } from '../../../../shared/models/videos/playlist/video-playlist-privacy.model'
|
||||||
|
|
||||||
const expect = chai.expect
|
async function testEndpoints (server: ServerInfo, token: string, filter: string, playlistUUID: string, statusCodeExpected: number) {
|
||||||
|
|
||||||
async function testEndpoints (server: ServerInfo, token: string, filter: string, statusCodeExpected: number) {
|
|
||||||
const paths = [
|
const paths = [
|
||||||
'/api/v1/video-channels/root_channel/videos',
|
'/api/v1/video-channels/root_channel/videos',
|
||||||
'/api/v1/accounts/root/videos',
|
'/api/v1/accounts/root/videos',
|
||||||
'/api/v1/videos',
|
'/api/v1/videos',
|
||||||
'/api/v1/search/videos'
|
'/api/v1/search/videos',
|
||||||
|
'/api/v1/video-playlists/' + playlistUUID + '/videos'
|
||||||
]
|
]
|
||||||
|
|
||||||
for (const path of paths) {
|
for (const path of paths) {
|
||||||
|
@ -41,6 +41,7 @@ describe('Test videos filters', function () {
|
||||||
let server: ServerInfo
|
let server: ServerInfo
|
||||||
let userAccessToken: string
|
let userAccessToken: string
|
||||||
let moderatorAccessToken: string
|
let moderatorAccessToken: string
|
||||||
|
let playlistUUID: string
|
||||||
|
|
||||||
// ---------------------------------------------------------------
|
// ---------------------------------------------------------------
|
||||||
|
|
||||||
|
@ -68,28 +69,38 @@ describe('Test videos filters', function () {
|
||||||
UserRole.MODERATOR
|
UserRole.MODERATOR
|
||||||
)
|
)
|
||||||
moderatorAccessToken = await userLogin(server, moderator)
|
moderatorAccessToken = await userLogin(server, moderator)
|
||||||
|
|
||||||
|
const res = await createVideoPlaylist({
|
||||||
|
url: server.url,
|
||||||
|
token: server.accessToken,
|
||||||
|
playlistAttrs: {
|
||||||
|
displayName: 'super playlist',
|
||||||
|
privacy: VideoPlaylistPrivacy.PUBLIC
|
||||||
|
}
|
||||||
|
})
|
||||||
|
playlistUUID = res.body.videoPlaylist.uuid
|
||||||
})
|
})
|
||||||
|
|
||||||
describe('When setting a video filter', function () {
|
describe('When setting a video filter', function () {
|
||||||
|
|
||||||
it('Should fail with a bad filter', async function () {
|
it('Should fail with a bad filter', async function () {
|
||||||
await testEndpoints(server, server.accessToken, 'bad-filter', 400)
|
await testEndpoints(server, server.accessToken, 'bad-filter', playlistUUID, 400)
|
||||||
})
|
})
|
||||||
|
|
||||||
it('Should succeed with a good filter', async function () {
|
it('Should succeed with a good filter', async function () {
|
||||||
await testEndpoints(server, server.accessToken,'local', 200)
|
await testEndpoints(server, server.accessToken,'local', playlistUUID, 200)
|
||||||
})
|
})
|
||||||
|
|
||||||
it('Should fail to list all-local with a simple user', async function () {
|
it('Should fail to list all-local with a simple user', async function () {
|
||||||
await testEndpoints(server, userAccessToken, 'all-local', 401)
|
await testEndpoints(server, userAccessToken, 'all-local', playlistUUID, 401)
|
||||||
})
|
})
|
||||||
|
|
||||||
it('Should succeed to list all-local with a moderator', async function () {
|
it('Should succeed to list all-local with a moderator', async function () {
|
||||||
await testEndpoints(server, moderatorAccessToken, 'all-local', 200)
|
await testEndpoints(server, moderatorAccessToken, 'all-local', playlistUUID, 200)
|
||||||
})
|
})
|
||||||
|
|
||||||
it('Should succeed to list all-local with an admin', async function () {
|
it('Should succeed to list all-local with an admin', async function () {
|
||||||
await testEndpoints(server, server.accessToken, 'all-local', 200)
|
await testEndpoints(server, server.accessToken, 'all-local', playlistUUID, 200)
|
||||||
})
|
})
|
||||||
|
|
||||||
// Because we cannot authenticate the user on the RSS endpoint
|
// Because we cannot authenticate the user on the RSS endpoint
|
||||||
|
@ -104,7 +115,7 @@ describe('Test videos filters', function () {
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
it('Should succed on the feeds endpoint with the local filter', async function () {
|
it('Should succeed on the feeds endpoint with the local filter', async function () {
|
||||||
await makeGetRequest({
|
await makeGetRequest({
|
||||||
url: server.url,
|
url: server.url,
|
||||||
path: '/feeds/videos.json',
|
path: '/feeds/videos.json',
|
||||||
|
|
|
@ -4,20 +4,26 @@ import * as chai from 'chai'
|
||||||
import 'mocha'
|
import 'mocha'
|
||||||
import { VideoDetails } from '../../../../shared/models/videos'
|
import { VideoDetails } from '../../../../shared/models/videos'
|
||||||
import {
|
import {
|
||||||
|
checkSegmentHash,
|
||||||
|
checkVideoFilesWereRemoved,
|
||||||
doubleFollow,
|
doubleFollow,
|
||||||
flushAndRunMultipleServers,
|
flushAndRunMultipleServers,
|
||||||
getFollowingListPaginationAndSort,
|
getFollowingListPaginationAndSort,
|
||||||
getVideo,
|
getVideo,
|
||||||
|
getVideoWithToken,
|
||||||
immutableAssign,
|
immutableAssign,
|
||||||
killallServers, makeGetRequest,
|
killallServers,
|
||||||
|
makeGetRequest,
|
||||||
|
removeVideo,
|
||||||
|
reRunServer,
|
||||||
root,
|
root,
|
||||||
ServerInfo,
|
ServerInfo,
|
||||||
setAccessTokensToServers, unfollow,
|
setAccessTokensToServers,
|
||||||
|
unfollow,
|
||||||
uploadVideo,
|
uploadVideo,
|
||||||
viewVideo,
|
viewVideo,
|
||||||
wait,
|
wait,
|
||||||
waitUntilLog,
|
waitUntilLog
|
||||||
checkVideoFilesWereRemoved, removeVideo, getVideoWithToken, reRunServer, checkSegmentHash
|
|
||||||
} from '../../../../shared/utils'
|
} from '../../../../shared/utils'
|
||||||
import { waitJobs } from '../../../../shared/utils/server/jobs'
|
import { waitJobs } from '../../../../shared/utils/server/jobs'
|
||||||
|
|
||||||
|
|
|
@ -2,10 +2,10 @@ import { VideoPlaylistPrivacy } from './video-playlist-privacy.model'
|
||||||
|
|
||||||
export interface VideoPlaylistCreate {
|
export interface VideoPlaylistCreate {
|
||||||
displayName: string
|
displayName: string
|
||||||
description: string
|
|
||||||
privacy: VideoPlaylistPrivacy
|
privacy: VideoPlaylistPrivacy
|
||||||
|
|
||||||
|
description?: string
|
||||||
videoChannelId?: number
|
videoChannelId?: number
|
||||||
|
|
||||||
thumbnailfile?: Blob
|
thumbnailfile?: any
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,4 +1,6 @@
|
||||||
export interface VideoPlaylistElementCreate {
|
export interface VideoPlaylistElementCreate {
|
||||||
|
videoId: number
|
||||||
|
|
||||||
startTimestamp?: number
|
startTimestamp?: number
|
||||||
stopTimestamp?: number
|
stopTimestamp?: number
|
||||||
}
|
}
|
||||||
|
|
|
@ -2,9 +2,9 @@ import { VideoPlaylistPrivacy } from './video-playlist-privacy.model'
|
||||||
|
|
||||||
export interface VideoPlaylistUpdate {
|
export interface VideoPlaylistUpdate {
|
||||||
displayName: string
|
displayName: string
|
||||||
description: string
|
|
||||||
privacy: VideoPlaylistPrivacy
|
privacy: VideoPlaylistPrivacy
|
||||||
|
|
||||||
|
description?: string
|
||||||
videoChannelId?: number
|
videoChannelId?: number
|
||||||
thumbnailfile?: Blob
|
thumbnailfile?: any
|
||||||
}
|
}
|
||||||
|
|
|
@ -13,12 +13,13 @@ export * from './requests/requests'
|
||||||
export * from './requests/check-api-params'
|
export * from './requests/check-api-params'
|
||||||
export * from './server/servers'
|
export * from './server/servers'
|
||||||
export * from './videos/services'
|
export * from './videos/services'
|
||||||
|
export * from './videos/video-playlists'
|
||||||
export * from './users/users'
|
export * from './users/users'
|
||||||
export * from './videos/video-abuses'
|
export * from './videos/video-abuses'
|
||||||
export * from './videos/video-blacklist'
|
export * from './videos/video-blacklist'
|
||||||
export * from './videos/video-channels'
|
export * from './videos/video-channels'
|
||||||
export * from './videos/video-comments'
|
export * from './videos/video-comments'
|
||||||
export * from './videos/video-playlists'
|
export * from './videos/video-streaming-playlists'
|
||||||
export * from './videos/videos'
|
export * from './videos/videos'
|
||||||
export * from './videos/video-change-ownership'
|
export * from './videos/video-change-ownership'
|
||||||
export * from './feeds/feeds'
|
export * from './feeds/feeds'
|
||||||
|
|
|
@ -31,7 +31,7 @@ function getVideoPlaylist (url: string, playlistId: number | string, statusCodeE
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
function deleteVideoPlaylist (url: string, token: string, playlistId: number | string, statusCodeExpected = 200) {
|
function deleteVideoPlaylist (url: string, token: string, playlistId: number | string, statusCodeExpected = 204) {
|
||||||
const path = '/api/v1/video-playlists/' + playlistId
|
const path = '/api/v1/video-playlists/' + playlistId
|
||||||
|
|
||||||
return makeDeleteRequest({
|
return makeDeleteRequest({
|
||||||
|
@ -46,7 +46,7 @@ function createVideoPlaylist (options: {
|
||||||
url: string,
|
url: string,
|
||||||
token: string,
|
token: string,
|
||||||
playlistAttrs: VideoPlaylistCreate,
|
playlistAttrs: VideoPlaylistCreate,
|
||||||
expectedStatus: number
|
expectedStatus?: number
|
||||||
}) {
|
}) {
|
||||||
const path = '/api/v1/video-playlists/'
|
const path = '/api/v1/video-playlists/'
|
||||||
|
|
||||||
|
@ -63,7 +63,7 @@ function createVideoPlaylist (options: {
|
||||||
token: options.token,
|
token: options.token,
|
||||||
fields,
|
fields,
|
||||||
attaches,
|
attaches,
|
||||||
statusCodeExpected: options.expectedStatus
|
statusCodeExpected: options.expectedStatus || 200
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -71,9 +71,10 @@ function updateVideoPlaylist (options: {
|
||||||
url: string,
|
url: string,
|
||||||
token: string,
|
token: string,
|
||||||
playlistAttrs: VideoPlaylistUpdate,
|
playlistAttrs: VideoPlaylistUpdate,
|
||||||
expectedStatus: number
|
playlistId: number | string,
|
||||||
|
expectedStatus?: number
|
||||||
}) {
|
}) {
|
||||||
const path = '/api/v1/video-playlists/'
|
const path = '/api/v1/video-playlists/' + options.playlistId
|
||||||
|
|
||||||
const fields = omit(options.playlistAttrs, 'thumbnailfile')
|
const fields = omit(options.playlistAttrs, 'thumbnailfile')
|
||||||
|
|
||||||
|
@ -88,7 +89,7 @@ function updateVideoPlaylist (options: {
|
||||||
token: options.token,
|
token: options.token,
|
||||||
fields,
|
fields,
|
||||||
attaches,
|
attaches,
|
||||||
statusCodeExpected: options.expectedStatus
|
statusCodeExpected: options.expectedStatus || 204
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -97,7 +98,7 @@ function addVideoInPlaylist (options: {
|
||||||
token: string,
|
token: string,
|
||||||
playlistId: number | string,
|
playlistId: number | string,
|
||||||
elementAttrs: VideoPlaylistElementCreate
|
elementAttrs: VideoPlaylistElementCreate
|
||||||
expectedStatus: number
|
expectedStatus?: number
|
||||||
}) {
|
}) {
|
||||||
const path = '/api/v1/video-playlists/' + options.playlistId + '/videos'
|
const path = '/api/v1/video-playlists/' + options.playlistId + '/videos'
|
||||||
|
|
||||||
|
@ -106,7 +107,7 @@ function addVideoInPlaylist (options: {
|
||||||
path,
|
path,
|
||||||
token: options.token,
|
token: options.token,
|
||||||
fields: options.elementAttrs,
|
fields: options.elementAttrs,
|
||||||
statusCodeExpected: options.expectedStatus
|
statusCodeExpected: options.expectedStatus || 200
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -116,7 +117,7 @@ function updateVideoPlaylistElement (options: {
|
||||||
playlistId: number | string,
|
playlistId: number | string,
|
||||||
videoId: number | string,
|
videoId: number | string,
|
||||||
elementAttrs: VideoPlaylistElementUpdate,
|
elementAttrs: VideoPlaylistElementUpdate,
|
||||||
expectedStatus: number
|
expectedStatus?: number
|
||||||
}) {
|
}) {
|
||||||
const path = '/api/v1/video-playlists/' + options.playlistId + '/videos/' + options.videoId
|
const path = '/api/v1/video-playlists/' + options.playlistId + '/videos/' + options.videoId
|
||||||
|
|
||||||
|
@ -125,7 +126,7 @@ function updateVideoPlaylistElement (options: {
|
||||||
path,
|
path,
|
||||||
token: options.token,
|
token: options.token,
|
||||||
fields: options.elementAttrs,
|
fields: options.elementAttrs,
|
||||||
statusCodeExpected: options.expectedStatus
|
statusCodeExpected: options.expectedStatus || 204
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -142,7 +143,7 @@ function removeVideoFromPlaylist (options: {
|
||||||
url: options.url,
|
url: options.url,
|
||||||
path,
|
path,
|
||||||
token: options.token,
|
token: options.token,
|
||||||
statusCodeExpected: options.expectedStatus
|
statusCodeExpected: options.expectedStatus || 204
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -152,14 +153,14 @@ function reorderVideosPlaylist (options: {
|
||||||
playlistId: number | string,
|
playlistId: number | string,
|
||||||
elementAttrs: {
|
elementAttrs: {
|
||||||
startPosition: number,
|
startPosition: number,
|
||||||
insertAfter: number,
|
insertAfterPosition: number,
|
||||||
reorderLength?: number
|
reorderLength?: number
|
||||||
},
|
},
|
||||||
expectedStatus: number
|
expectedStatus: number
|
||||||
}) {
|
}) {
|
||||||
const path = '/api/v1/video-playlists/' + options.playlistId + '/videos'
|
const path = '/api/v1/video-playlists/' + options.playlistId + '/videos/reorder'
|
||||||
|
|
||||||
return makePutBodyRequest({
|
return makePostBodyRequest({
|
||||||
url: options.url,
|
url: options.url,
|
||||||
path,
|
path,
|
||||||
token: options.token,
|
token: options.token,
|
||||||
|
@ -179,6 +180,7 @@ export {
|
||||||
deleteVideoPlaylist,
|
deleteVideoPlaylist,
|
||||||
|
|
||||||
addVideoInPlaylist,
|
addVideoInPlaylist,
|
||||||
|
updateVideoPlaylistElement,
|
||||||
removeVideoFromPlaylist,
|
removeVideoFromPlaylist,
|
||||||
|
|
||||||
reorderVideosPlaylist
|
reorderVideosPlaylist
|
||||||
|
|
Loading…
Reference in New Issue