+
+ it('Should not list comments from muted accounts or instances', async function () {
+ this.timeout(30000)
+
+ const remoteHandle = 'root@' + servers[0].host
+
+ await servers[1].blocklist.addToServerBlocklist({ account: remoteHandle })
+
+ {
+ const json = await servers[1].feed.getJSON({ feed: 'video-comments', ignoreCache: true })
+ const jsonObj = JSON.parse(json)
+ expect(jsonObj.items.length).to.be.equal(0)
+ }
+
+ await servers[1].blocklist.removeFromServerBlocklist({ account: remoteHandle })
+
+ {
+ const videoUUID = (await servers[1].videos.quickUpload({ name: 'server 2' })).uuid
+ await waitJobs(servers)
+ await servers[0].comments.createThread({ videoId: videoUUID, text: 'super comment' })
+ await waitJobs(servers)
+
+ const json = await servers[1].feed.getJSON({ feed: 'video-comments', ignoreCache: true })
+ const jsonObj = JSON.parse(json)
+ expect(jsonObj.items.length).to.be.equal(3)
+ }
+
+ await servers[1].blocklist.addToMyBlocklist({ account: remoteHandle })
+
+ {
+ const json = await servers[1].feed.getJSON({ feed: 'video-comments', ignoreCache: true })
+ const jsonObj = JSON.parse(json)
+ expect(jsonObj.items.length).to.be.equal(2)
+ }
+ })
+ })
+
+ describe('Video feed from my subscriptions', function () {
+ let feeduserAccountId: number
+ let feeduserFeedToken: string
+
+ it('Should list no videos for a user with no videos and no subscriptions', async function () {
+ const attr = { username: 'feeduser', password: 'password' }
+ await servers[0].users.create({ username: attr.username, password: attr.password })
+ const feeduserAccessToken = await servers[0].login.getAccessToken(attr)
+
+ {
+ const user = await servers[0].users.getMyInfo({ token: feeduserAccessToken })
+ feeduserAccountId = user.account.id
+ }
+
+ {
+ const token = await servers[0].users.getMyScopedTokens({ token: feeduserAccessToken })
+ feeduserFeedToken = token.feedToken
+ }
+
+ {
+ const body = await servers[0].videos.listMySubscriptionVideos({ token: feeduserAccessToken })
+ expect(body.total).to.equal(0)
+
+ const query = { accountId: feeduserAccountId, token: feeduserFeedToken }
+ const json = await servers[0].feed.getJSON({ feed: 'subscriptions', query, ignoreCache: true })
+ const jsonObj = JSON.parse(json)
+ expect(jsonObj.items.length).to.be.equal(0) // no subscription, it should not list the instance's videos but list 0 videos
+ }
+ })
+
+ it('Should fail with an invalid token', async function () {
+ const query = { accountId: feeduserAccountId, token: 'toto' }
+ await servers[0].feed.getJSON({ feed: 'subscriptions', query, expectedStatus: HttpStatusCode.FORBIDDEN_403, ignoreCache: true })
+ })
+
+ it('Should fail with a token of another user', async function () {
+ const query = { accountId: feeduserAccountId, token: userFeedToken }
+ await servers[0].feed.getJSON({ feed: 'subscriptions', query, expectedStatus: HttpStatusCode.FORBIDDEN_403, ignoreCache: true })
+ })
+
+ it('Should list no videos for a user with videos but no subscriptions', async function () {
+ const body = await servers[0].videos.listMySubscriptionVideos({ token: userAccessToken })
+ expect(body.total).to.equal(0)
+
+ const query = { accountId: userAccountId, token: userFeedToken }
+ const json = await servers[0].feed.getJSON({ feed: 'subscriptions', query, ignoreCache: true })
+ const jsonObj = JSON.parse(json)
+ expect(jsonObj.items.length).to.be.equal(0) // no subscription, it should not list the instance's videos but list 0 videos
+ })
+
+ it('Should list self videos for a user with a subscription to themselves', async function () {
+ this.timeout(30000)
+
+ await servers[0].subscriptions.add({ token: userAccessToken, targetUri: 'john_channel@' + servers[0].host })
+ await waitJobs(servers)
+
+ {
+ const body = await servers[0].videos.listMySubscriptionVideos({ token: userAccessToken })
+ expect(body.total).to.equal(1)
+ expect(body.data[0].name).to.equal('user video')
+
+ const query = { accountId: userAccountId, token: userFeedToken }
+ const json = await servers[0].feed.getJSON({ feed: 'subscriptions', query, ignoreCache: true })
+ const jsonObj = JSON.parse(json)
+ expect(jsonObj.items.length).to.be.equal(1) // subscribed to self, it should not list the instance's videos but list john's
+ }
+ })
+
+ it('Should list videos of a user\'s subscription', async function () {
+ this.timeout(30000)
+
+ await servers[0].subscriptions.add({ token: userAccessToken, targetUri: 'root_channel@' + servers[0].host })
+ await waitJobs(servers)
+
+ {
+ const body = await servers[0].videos.listMySubscriptionVideos({ token: userAccessToken })
+ expect(body.total).to.equal(2, 'there should be 2 videos part of the subscription')
+
+ const query = { accountId: userAccountId, token: userFeedToken }
+ const json = await servers[0].feed.getJSON({ feed: 'subscriptions', query, ignoreCache: true })
+ const jsonObj = JSON.parse(json)
+ expect(jsonObj.items.length).to.be.equal(2) // subscribed to root, it should not list the instance's videos but list root/john's
+ }
+ })
+
+ it('Should renew the token, and so have an invalid old token', async function () {
+ await servers[0].users.renewMyScopedTokens({ token: userAccessToken })
+
+ const query = { accountId: userAccountId, token: userFeedToken }
+ await servers[0].feed.getJSON({ feed: 'subscriptions', query, expectedStatus: HttpStatusCode.FORBIDDEN_403, ignoreCache: true })
+ })
+
+ it('Should succeed with the new token', async function () {
+ const token = await servers[0].users.getMyScopedTokens({ token: userAccessToken })
+ userFeedToken = token.feedToken
+
+ const query = { accountId: userAccountId, token: userFeedToken }
+ await servers[0].feed.getJSON({ feed: 'subscriptions', query, ignoreCache: true })
+ })
+
+ })
+
+ describe('Cache', function () {
+ const uuids: string[] = []
+
+ function doPodcastRequest () {
+ return makeGetRequest({
+ url: servers[0].url,
+ path: '/feeds/podcast/videos.xml',
+ query: { videoChannelId: servers[0].store.channel.id },
+ accept: 'application/xml',
+ expectedStatus: HttpStatusCode.OK_200
+ })
+ }
+
+ function doVideosRequest (query: { [id: string]: string } = {}) {
+ return makeGetRequest({
+ url: servers[0].url,
+ path: '/feeds/videos.xml',
+ query,
+ accept: 'application/xml',
+ expectedStatus: HttpStatusCode.OK_200
+ })
+ }
+
+ before(async function () {
+ {
+ const { uuid } = await servers[0].videos.quickUpload({ name: 'cache 1' })
+ uuids.push(uuid)
+ }
+
+ {
+ const { uuid } = await servers[0].videos.quickUpload({ name: 'cache 2' })
+ uuids.push(uuid)
+ }
+ })
+
+ it('Should serve the videos endpoint as a cached request', async function () {
+ await doVideosRequest()
+
+ const res = await doVideosRequest()
+
+ expect(res.headers['x-api-cache-cached']).to.equal('true')
+ })
+
+ it('Should not serve the videos endpoint as a cached request', async function () {
+ const res = await doVideosRequest({ v: '186' })
+
+ expect(res.headers['x-api-cache-cached']).to.not.exist
+ })
+
+ it('Should invalidate the podcast feed cache after video deletion', async function () {
+ await doPodcastRequest()
+
+ {
+ const res = await doPodcastRequest()
+ expect(res.headers['x-api-cache-cached']).to.exist
+ }
+
+ await servers[0].videos.remove({ id: uuids[0] })
+
+ {
+ const res = await doPodcastRequest()
+ expect(res.headers['x-api-cache-cached']).to.not.exist
+ }
+ })
+
+ it('Should invalidate the podcast feed cache after video deletion, even after server restart', async function () {
+ this.timeout(120000)
+
+ await doPodcastRequest()
+
+ {
+ const res = await doPodcastRequest()
+ expect(res.headers['x-api-cache-cached']).to.exist
+ }
+
+ await servers[0].kill()
+ await servers[0].run()
+
+ await servers[0].videos.remove({ id: uuids[1] })
+
+ const res = await doPodcastRequest()
+ expect(res.headers['x-api-cache-cached']).to.not.exist
+ })
+