123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551 |
- /* eslint-disable @typescript-eslint/no-unused-expressions,@typescript-eslint/require-await */
- import { expect } from 'chai'
- import { basename } from 'path'
- import { ACTOR_IMAGES_SIZE } from '@server/initializers/constants'
- import { testFileExistsOrNot, testImage } from '@server/tests/shared'
- import { wait } from '@shared/core-utils'
- import { ActorImageType, User, VideoChannel } from '@shared/models'
- import {
- cleanupTests,
- createMultipleServers,
- doubleFollow,
- PeerTubeServer,
- setAccessTokensToServers,
- setDefaultAccountAvatar,
- setDefaultVideoChannel,
- waitJobs
- } from '@shared/server-commands'
- async function findChannel (server: PeerTubeServer, channelId: number) {
- const body = await server.channels.list({ sort: '-name' })
- return body.data.find(c => c.id === channelId)
- }
- describe('Test video channels', function () {
- let servers: PeerTubeServer[]
- let userInfo: User
- let secondVideoChannelId: number
- let totoChannel: number
- let videoUUID: string
- let accountName: string
- let secondUserChannelName: string
- const avatarPaths: { [ port: number ]: string } = {}
- const bannerPaths: { [ port: number ]: string } = {}
- before(async function () {
- this.timeout(60000)
- servers = await createMultipleServers(2)
- await setAccessTokensToServers(servers)
- await setDefaultVideoChannel(servers)
- await setDefaultAccountAvatar(servers)
- await doubleFollow(servers[0], servers[1])
- })
- it('Should have one video channel (created with root)', async () => {
- const body = await servers[0].channels.list({ start: 0, count: 2 })
- expect(body.total).to.equal(1)
- expect(body.data).to.be.an('array')
- expect(body.data).to.have.lengthOf(1)
- })
- it('Should create another video channel', async function () {
- this.timeout(30000)
- {
- const videoChannel = {
- name: 'second_video_channel',
- displayName: 'second video channel',
- description: 'super video channel description',
- support: 'super video channel support text'
- }
- const created = await servers[0].channels.create({ attributes: videoChannel })
- secondVideoChannelId = created.id
- }
- // The channel is 1 is propagated to servers 2
- {
- const attributes = { name: 'my video name', channelId: secondVideoChannelId, support: 'video support field' }
- const { uuid } = await servers[0].videos.upload({ attributes })
- videoUUID = uuid
- }
- await waitJobs(servers)
- })
- it('Should have two video channels when getting my information', async () => {
- userInfo = await servers[0].users.getMyInfo()
- expect(userInfo.videoChannels).to.be.an('array')
- expect(userInfo.videoChannels).to.have.lengthOf(2)
- const videoChannels = userInfo.videoChannels
- expect(videoChannels[0].name).to.equal('root_channel')
- expect(videoChannels[0].displayName).to.equal('Main root channel')
- expect(videoChannels[1].name).to.equal('second_video_channel')
- expect(videoChannels[1].displayName).to.equal('second video channel')
- expect(videoChannels[1].description).to.equal('super video channel description')
- expect(videoChannels[1].support).to.equal('super video channel support text')
- accountName = userInfo.account.name + '@' + userInfo.account.host
- })
- it('Should have two video channels when getting account channels on server 1', async function () {
- const body = await servers[0].channels.listByAccount({ accountName })
- expect(body.total).to.equal(2)
- const videoChannels = body.data
- expect(videoChannels).to.be.an('array')
- expect(videoChannels).to.have.lengthOf(2)
- expect(videoChannels[0].name).to.equal('root_channel')
- expect(videoChannels[0].displayName).to.equal('Main root channel')
- expect(videoChannels[1].name).to.equal('second_video_channel')
- expect(videoChannels[1].displayName).to.equal('second video channel')
- expect(videoChannels[1].description).to.equal('super video channel description')
- expect(videoChannels[1].support).to.equal('super video channel support text')
- })
- it('Should paginate and sort account channels', async function () {
- {
- const body = await servers[0].channels.listByAccount({
- accountName,
- start: 0,
- count: 1,
- sort: 'createdAt'
- })
- expect(body.total).to.equal(2)
- expect(body.data).to.have.lengthOf(1)
- const videoChannel: VideoChannel = body.data[0]
- expect(videoChannel.name).to.equal('root_channel')
- }
- {
- const body = await servers[0].channels.listByAccount({
- accountName,
- start: 0,
- count: 1,
- sort: '-createdAt'
- })
- expect(body.total).to.equal(2)
- expect(body.data).to.have.lengthOf(1)
- expect(body.data[0].name).to.equal('second_video_channel')
- }
- {
- const body = await servers[0].channels.listByAccount({
- accountName,
- start: 1,
- count: 1,
- sort: '-createdAt'
- })
- expect(body.total).to.equal(2)
- expect(body.data).to.have.lengthOf(1)
- expect(body.data[0].name).to.equal('root_channel')
- }
- })
- it('Should have one video channel when getting account channels on server 2', async function () {
- const body = await servers[1].channels.listByAccount({ accountName })
- expect(body.total).to.equal(1)
- expect(body.data).to.be.an('array')
- expect(body.data).to.have.lengthOf(1)
- const videoChannel = body.data[0]
- expect(videoChannel.name).to.equal('second_video_channel')
- expect(videoChannel.displayName).to.equal('second video channel')
- expect(videoChannel.description).to.equal('super video channel description')
- expect(videoChannel.support).to.equal('super video channel support text')
- })
- it('Should list video channels', async function () {
- const body = await servers[0].channels.list({ start: 1, count: 1, sort: '-name' })
- expect(body.total).to.equal(2)
- expect(body.data).to.be.an('array')
- expect(body.data).to.have.lengthOf(1)
- expect(body.data[0].name).to.equal('root_channel')
- expect(body.data[0].displayName).to.equal('Main root channel')
- })
- it('Should update video channel', async function () {
- this.timeout(15000)
- const videoChannelAttributes = {
- displayName: 'video channel updated',
- description: 'video channel description updated',
- support: 'support updated'
- }
- await servers[0].channels.update({ channelName: 'second_video_channel', attributes: videoChannelAttributes })
- await waitJobs(servers)
- })
- it('Should have video channel updated', async function () {
- for (const server of servers) {
- const body = await server.channels.list({ start: 0, count: 1, sort: '-name' })
- expect(body.total).to.equal(2)
- expect(body.data).to.be.an('array')
- expect(body.data).to.have.lengthOf(1)
- expect(body.data[0].name).to.equal('second_video_channel')
- expect(body.data[0].displayName).to.equal('video channel updated')
- expect(body.data[0].description).to.equal('video channel description updated')
- expect(body.data[0].support).to.equal('support updated')
- }
- })
- it('Should not have updated the video support field', async function () {
- for (const server of servers) {
- const video = await server.videos.get({ id: videoUUID })
- expect(video.support).to.equal('video support field')
- }
- })
- it('Should update another accounts video channel', async function () {
- this.timeout(15000)
- const result = await servers[0].users.generate('second_user')
- secondUserChannelName = result.userChannelName
- await servers[0].videos.quickUpload({ name: 'video', token: result.token })
- const videoChannelAttributes = {
- displayName: 'video channel updated',
- description: 'video channel description updated',
- support: 'support updated'
- }
- await servers[0].channels.update({ channelName: secondUserChannelName, attributes: videoChannelAttributes })
- await waitJobs(servers)
- })
- it('Should have another accounts video channel updated', async function () {
- for (const server of servers) {
- const body = await server.channels.get({ channelName: `${secondUserChannelName}@${servers[0].host}` })
- expect(body.displayName).to.equal('video channel updated')
- expect(body.description).to.equal('video channel description updated')
- expect(body.support).to.equal('support updated')
- }
- })
- it('Should update the channel support field and update videos too', async function () {
- this.timeout(35000)
- const videoChannelAttributes = {
- support: 'video channel support text updated',
- bulkVideosSupportUpdate: true
- }
- await servers[0].channels.update({ channelName: 'second_video_channel', attributes: videoChannelAttributes })
- await waitJobs(servers)
- for (const server of servers) {
- const video = await server.videos.get({ id: videoUUID })
- expect(video.support).to.equal(videoChannelAttributes.support)
- }
- })
- it('Should update video channel avatar', async function () {
- this.timeout(15000)
- const fixture = 'avatar.png'
- await servers[0].channels.updateImage({
- channelName: 'second_video_channel',
- fixture,
- type: 'avatar'
- })
- await waitJobs(servers)
- for (const server of servers) {
- const videoChannel = await findChannel(server, secondVideoChannelId)
- const expectedSizes = ACTOR_IMAGES_SIZE[ActorImageType.AVATAR]
- expect(videoChannel.avatars.length).to.equal(expectedSizes.length, 'Expected avatars to be generated in all sizes')
- for (const avatar of videoChannel.avatars) {
- avatarPaths[server.port] = avatar.path
- await testImage(server.url, `avatar-resized-${avatar.width}x${avatar.width}`, avatarPaths[server.port], '.png')
- await testFileExistsOrNot(server, 'avatars', basename(avatarPaths[server.port]), true)
- const row = await server.sql.getActorImage(basename(avatarPaths[server.port]))
- expect(expectedSizes.some(({ height, width }) => row.height === height && row.width === width)).to.equal(true)
- }
- }
- })
- it('Should update video channel banner', async function () {
- this.timeout(15000)
- const fixture = 'banner.jpg'
- await servers[0].channels.updateImage({
- channelName: 'second_video_channel',
- fixture,
- type: 'banner'
- })
- await waitJobs(servers)
- for (const server of servers) {
- const videoChannel = await server.channels.get({ channelName: 'second_video_channel@' + servers[0].host })
- bannerPaths[server.port] = videoChannel.banners[0].path
- await testImage(server.url, 'banner-resized', bannerPaths[server.port])
- await testFileExistsOrNot(server, 'avatars', basename(bannerPaths[server.port]), true)
- const row = await server.sql.getActorImage(basename(bannerPaths[server.port]))
- expect(row.height).to.equal(ACTOR_IMAGES_SIZE[ActorImageType.BANNER][0].height)
- expect(row.width).to.equal(ACTOR_IMAGES_SIZE[ActorImageType.BANNER][0].width)
- }
- })
- it('Should still correctly list channels', async function () {
- {
- const body = await servers[0].channels.list({ start: 1, count: 1, sort: 'createdAt' })
- expect(body.total).to.equal(3)
- expect(body.data).to.have.lengthOf(1)
- expect(body.data[0].name).to.equal('second_video_channel')
- }
- {
- const body = await servers[0].channels.listByAccount({ accountName, start: 1, count: 1, sort: 'createdAt' })
- expect(body.total).to.equal(2)
- expect(body.data).to.have.lengthOf(1)
- expect(body.data[0].name).to.equal('second_video_channel')
- }
- })
- it('Should delete the video channel avatar', async function () {
- this.timeout(15000)
- await servers[0].channels.deleteImage({ channelName: 'second_video_channel', type: 'avatar' })
- await waitJobs(servers)
- for (const server of servers) {
- const videoChannel = await findChannel(server, secondVideoChannelId)
- await testFileExistsOrNot(server, 'avatars', basename(avatarPaths[server.port]), false)
- expect(videoChannel.avatars).to.be.empty
- }
- })
- it('Should delete the video channel banner', async function () {
- this.timeout(15000)
- await servers[0].channels.deleteImage({ channelName: 'second_video_channel', type: 'banner' })
- await waitJobs(servers)
- for (const server of servers) {
- const videoChannel = await findChannel(server, secondVideoChannelId)
- await testFileExistsOrNot(server, 'avatars', basename(bannerPaths[server.port]), false)
- expect(videoChannel.banners).to.be.empty
- }
- })
- it('Should list the second video channel videos', async function () {
- this.timeout(10000)
- for (const server of servers) {
- const channelURI = 'second_video_channel@localhost:' + servers[0].port
- const { total, data } = await server.videos.listByChannel({ handle: channelURI })
- expect(total).to.equal(1)
- expect(data).to.be.an('array')
- expect(data).to.have.lengthOf(1)
- expect(data[0].name).to.equal('my video name')
- }
- })
- it('Should change the video channel of a video', async function () {
- this.timeout(10000)
- await servers[0].videos.update({ id: videoUUID, attributes: { channelId: servers[0].store.channel.id } })
- await waitJobs(servers)
- })
- it('Should list the first video channel videos', async function () {
- this.timeout(10000)
- for (const server of servers) {
- {
- const secondChannelURI = 'second_video_channel@localhost:' + servers[0].port
- const { total } = await server.videos.listByChannel({ handle: secondChannelURI })
- expect(total).to.equal(0)
- }
- {
- const channelURI = 'root_channel@localhost:' + servers[0].port
- const { total, data } = await server.videos.listByChannel({ handle: channelURI })
- expect(total).to.equal(1)
- expect(data).to.be.an('array')
- expect(data).to.have.lengthOf(1)
- expect(data[0].name).to.equal('my video name')
- }
- }
- })
- it('Should delete video channel', async function () {
- await servers[0].channels.delete({ channelName: 'second_video_channel' })
- })
- it('Should have video channel deleted', async function () {
- const body = await servers[0].channels.list({ start: 0, count: 10, sort: 'createdAt' })
- expect(body.total).to.equal(2)
- expect(body.data).to.be.an('array')
- expect(body.data).to.have.lengthOf(2)
- expect(body.data[0].displayName).to.equal('Main root channel')
- expect(body.data[1].displayName).to.equal('video channel updated')
- })
- it('Should create the main channel with a suffix if there is a conflict', async function () {
- {
- const videoChannel = { name: 'toto_channel', displayName: 'My toto channel' }
- const created = await servers[0].channels.create({ attributes: videoChannel })
- totoChannel = created.id
- }
- {
- await servers[0].users.create({ username: 'toto', password: 'password' })
- const accessToken = await servers[0].login.getAccessToken({ username: 'toto', password: 'password' })
- const { videoChannels } = await servers[0].users.getMyInfo({ token: accessToken })
- expect(videoChannels[0].name).to.equal('toto_channel-1')
- }
- })
- it('Should report correct channel views per days', async function () {
- this.timeout(10000)
- {
- const { data } = await servers[0].channels.listByAccount({ accountName, withStats: true })
- for (const channel of data) {
- expect(channel).to.haveOwnProperty('viewsPerDay')
- expect(channel.viewsPerDay).to.have.length(30 + 1) // daysPrior + today
- for (const v of channel.viewsPerDay) {
- expect(v.date).to.be.an('string')
- expect(v.views).to.equal(0)
- }
- }
- }
- {
- // video has been posted on channel servers[0].store.videoChannel.id since last update
- await servers[0].views.simulateView({ id: videoUUID, xForwardedFor: '0.0.0.1,127.0.0.1' })
- await servers[0].views.simulateView({ id: videoUUID, xForwardedFor: '0.0.0.2,127.0.0.1' })
- // Wait the repeatable job
- await wait(8000)
- const { data } = await servers[0].channels.listByAccount({ accountName, withStats: true })
- const channelWithView = data.find(channel => channel.id === servers[0].store.channel.id)
- expect(channelWithView.viewsPerDay.slice(-1)[0].views).to.equal(2)
- }
- })
- it('Should report correct total views count', async function () {
- // check if there's the property
- {
- const { data } = await servers[0].channels.listByAccount({ accountName, withStats: true })
- for (const channel of data) {
- expect(channel).to.haveOwnProperty('totalViews')
- expect(channel.totalViews).to.be.a('number')
- }
- }
- // Check if the totalViews count can be updated
- {
- const { data } = await servers[0].channels.listByAccount({ accountName, withStats: true })
- const channelWithView = data.find(channel => channel.id === servers[0].store.channel.id)
- expect(channelWithView.totalViews).to.equal(2)
- }
- })
- it('Should report correct videos count', async function () {
- const { data } = await servers[0].channels.listByAccount({ accountName, withStats: true })
- const totoChannel = data.find(c => c.name === 'toto_channel')
- const rootChannel = data.find(c => c.name === 'root_channel')
- expect(rootChannel.videosCount).to.equal(1)
- expect(totoChannel.videosCount).to.equal(0)
- })
- it('Should search among account video channels', async function () {
- {
- const body = await servers[0].channels.listByAccount({ accountName, search: 'root' })
- expect(body.total).to.equal(1)
- const channels = body.data
- expect(channels).to.have.lengthOf(1)
- }
- {
- const body = await servers[0].channels.listByAccount({ accountName, search: 'does not exist' })
- expect(body.total).to.equal(0)
- const channels = body.data
- expect(channels).to.have.lengthOf(0)
- }
- })
- it('Should list channels by updatedAt desc if a video has been uploaded', async function () {
- this.timeout(30000)
- await servers[0].videos.upload({ attributes: { channelId: totoChannel } })
- await waitJobs(servers)
- for (const server of servers) {
- const { data } = await server.channels.listByAccount({ accountName, sort: '-updatedAt' })
- expect(data[0].name).to.equal('toto_channel')
- expect(data[1].name).to.equal('root_channel')
- }
- await servers[0].videos.upload({ attributes: { channelId: servers[0].store.channel.id } })
- await waitJobs(servers)
- for (const server of servers) {
- const { data } = await server.channels.listByAccount({ accountName, sort: '-updatedAt' })
- expect(data[0].name).to.equal('root_channel')
- expect(data[1].name).to.equal('toto_channel')
- }
- })
- after(async function () {
- await cleanupTests(servers)
- })
- })
|