2022-03-24 13:36:47 +01:00
|
|
|
/* eslint-disable @typescript-eslint/no-unused-expressions,@typescript-eslint/require-await */
|
|
|
|
|
2022-08-17 15:44:32 +02:00
|
|
|
import { expect } from 'chai'
|
2022-03-24 13:36:47 +01:00
|
|
|
import { FfmpegCommand } from 'fluent-ffmpeg'
|
|
|
|
import { prepareViewsServers, prepareViewsVideos, processViewersStats } from '@server/tests/shared'
|
|
|
|
import { VideoStatsTimeserie, VideoStatsTimeserieMetric } from '@shared/models'
|
|
|
|
import { cleanupTests, PeerTubeServer, stopFfmpeg } from '@shared/server-commands'
|
|
|
|
|
2022-05-06 14:23:02 +02:00
|
|
|
function buildOneMonthAgo () {
|
|
|
|
const monthAgo = new Date()
|
|
|
|
monthAgo.setHours(0, 0, 0, 0)
|
|
|
|
|
|
|
|
monthAgo.setDate(monthAgo.getDate() - 29)
|
|
|
|
|
|
|
|
return monthAgo
|
|
|
|
}
|
|
|
|
|
2022-03-24 13:36:47 +01:00
|
|
|
describe('Test views timeserie stats', function () {
|
|
|
|
const availableMetrics: VideoStatsTimeserieMetric[] = [ 'viewers' ]
|
|
|
|
|
|
|
|
let servers: PeerTubeServer[]
|
|
|
|
|
|
|
|
before(async function () {
|
|
|
|
this.timeout(120000)
|
|
|
|
|
|
|
|
servers = await prepareViewsServers()
|
|
|
|
})
|
|
|
|
|
|
|
|
describe('Common metric tests', function () {
|
|
|
|
let vodVideoId: string
|
|
|
|
|
|
|
|
before(async function () {
|
2022-11-15 08:55:27 +01:00
|
|
|
this.timeout(240000);
|
2022-03-24 13:36:47 +01:00
|
|
|
|
|
|
|
({ vodVideoId } = await prepareViewsVideos({ servers, live: false, vod: true }))
|
|
|
|
})
|
|
|
|
|
|
|
|
it('Should display empty metric stats', async function () {
|
|
|
|
for (const metric of availableMetrics) {
|
|
|
|
const { data } = await servers[0].videoStats.getTimeserieStats({ videoId: vodVideoId, metric })
|
|
|
|
|
2022-05-06 14:23:02 +02:00
|
|
|
expect(data).to.have.length.at.least(1)
|
2022-03-24 13:36:47 +01:00
|
|
|
|
|
|
|
for (const d of data) {
|
|
|
|
expect(d.value).to.equal(0)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
describe('Test viewer and watch time metrics on live and VOD', function () {
|
|
|
|
let vodVideoId: string
|
|
|
|
let liveVideoId: string
|
|
|
|
let command: FfmpegCommand
|
|
|
|
|
2022-05-06 14:23:02 +02:00
|
|
|
function expectTodayLastValue (result: VideoStatsTimeserie, lastValue?: number) {
|
2022-03-24 13:36:47 +01:00
|
|
|
const { data } = result
|
|
|
|
|
|
|
|
const last = data[data.length - 1]
|
|
|
|
const today = new Date().getDate()
|
|
|
|
expect(new Date(last.date).getDate()).to.equal(today)
|
2022-05-06 14:23:02 +02:00
|
|
|
|
|
|
|
if (lastValue) expect(last.value).to.equal(lastValue)
|
2022-04-07 10:53:35 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
function expectTimeserieData (result: VideoStatsTimeserie, lastValue: number) {
|
|
|
|
const { data } = result
|
2022-05-06 14:23:02 +02:00
|
|
|
expect(data).to.have.length.at.least(25)
|
2022-04-07 10:53:35 +02:00
|
|
|
|
|
|
|
expectTodayLastValue(result, lastValue)
|
2022-03-24 13:36:47 +01:00
|
|
|
|
|
|
|
for (let i = 0; i < data.length - 2; i++) {
|
|
|
|
expect(data[i].value).to.equal(0)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-07 10:53:35 +02:00
|
|
|
function expectInterval (result: VideoStatsTimeserie, intervalMs: number) {
|
|
|
|
const first = result.data[0]
|
|
|
|
const second = result.data[1]
|
|
|
|
expect(new Date(second.date).getTime() - new Date(first.date).getTime()).to.equal(intervalMs)
|
|
|
|
}
|
|
|
|
|
2022-03-24 13:36:47 +01:00
|
|
|
before(async function () {
|
2022-11-15 08:55:27 +01:00
|
|
|
this.timeout(240000);
|
2022-03-24 13:36:47 +01:00
|
|
|
|
|
|
|
({ vodVideoId, liveVideoId, ffmpegCommand: command } = await prepareViewsVideos({ servers, live: true, vod: true }))
|
|
|
|
})
|
|
|
|
|
|
|
|
it('Should display appropriate viewers metrics', async function () {
|
|
|
|
for (const videoId of [ vodVideoId, liveVideoId ]) {
|
|
|
|
await servers[0].views.simulateViewer({ id: videoId, currentTimes: [ 0, 3 ] })
|
|
|
|
await servers[1].views.simulateViewer({ id: videoId, currentTimes: [ 0, 5 ] })
|
|
|
|
}
|
|
|
|
|
|
|
|
await processViewersStats(servers)
|
|
|
|
|
|
|
|
for (const videoId of [ vodVideoId, liveVideoId ]) {
|
2022-05-06 14:23:02 +02:00
|
|
|
const result = await servers[0].videoStats.getTimeserieStats({
|
|
|
|
videoId,
|
|
|
|
startDate: buildOneMonthAgo(),
|
|
|
|
endDate: new Date(),
|
|
|
|
metric: 'viewers'
|
|
|
|
})
|
2022-03-24 13:36:47 +01:00
|
|
|
expectTimeserieData(result, 2)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
it('Should display appropriate watch time metrics', async function () {
|
|
|
|
for (const videoId of [ vodVideoId, liveVideoId ]) {
|
2022-05-06 14:23:02 +02:00
|
|
|
const result = await servers[0].videoStats.getTimeserieStats({
|
|
|
|
videoId,
|
|
|
|
startDate: buildOneMonthAgo(),
|
|
|
|
endDate: new Date(),
|
|
|
|
metric: 'aggregateWatchTime'
|
|
|
|
})
|
2022-03-24 13:36:47 +01:00
|
|
|
expectTimeserieData(result, 8)
|
|
|
|
|
|
|
|
await servers[1].views.simulateViewer({ id: videoId, currentTimes: [ 0, 1 ] })
|
|
|
|
}
|
|
|
|
|
|
|
|
await processViewersStats(servers)
|
|
|
|
|
|
|
|
for (const videoId of [ vodVideoId, liveVideoId ]) {
|
2022-05-06 14:23:02 +02:00
|
|
|
const result = await servers[0].videoStats.getTimeserieStats({
|
|
|
|
videoId,
|
|
|
|
startDate: buildOneMonthAgo(),
|
|
|
|
endDate: new Date(),
|
|
|
|
metric: 'aggregateWatchTime'
|
|
|
|
})
|
2022-03-24 13:36:47 +01:00
|
|
|
expectTimeserieData(result, 9)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
2022-04-07 10:53:35 +02:00
|
|
|
it('Should use a custom start/end date', async function () {
|
|
|
|
const now = new Date()
|
2022-04-08 10:22:56 +02:00
|
|
|
const twentyDaysAgo = new Date()
|
|
|
|
twentyDaysAgo.setDate(twentyDaysAgo.getDate() - 19)
|
2022-04-07 10:53:35 +02:00
|
|
|
|
|
|
|
const result = await servers[0].videoStats.getTimeserieStats({
|
|
|
|
videoId: vodVideoId,
|
|
|
|
metric: 'aggregateWatchTime',
|
2022-04-08 10:22:56 +02:00
|
|
|
startDate: twentyDaysAgo,
|
2022-04-07 10:53:35 +02:00
|
|
|
endDate: now
|
|
|
|
})
|
|
|
|
|
2022-04-08 10:22:56 +02:00
|
|
|
expect(result.groupInterval).to.equal('1 day')
|
|
|
|
expect(result.data).to.have.lengthOf(20)
|
2022-04-07 10:53:35 +02:00
|
|
|
|
|
|
|
const first = result.data[0]
|
2022-04-08 10:22:56 +02:00
|
|
|
expect(new Date(first.date).toLocaleDateString()).to.equal(twentyDaysAgo.toLocaleDateString())
|
2022-04-07 10:53:35 +02:00
|
|
|
|
|
|
|
expectInterval(result, 24 * 3600 * 1000)
|
|
|
|
expectTodayLastValue(result, 9)
|
|
|
|
})
|
|
|
|
|
2022-05-06 14:23:02 +02:00
|
|
|
it('Should automatically group by months', async function () {
|
|
|
|
const now = new Date()
|
|
|
|
const heightYearsAgo = new Date()
|
|
|
|
heightYearsAgo.setFullYear(heightYearsAgo.getFullYear() - 7)
|
|
|
|
|
|
|
|
const result = await servers[0].videoStats.getTimeserieStats({
|
|
|
|
videoId: vodVideoId,
|
|
|
|
metric: 'aggregateWatchTime',
|
|
|
|
startDate: heightYearsAgo,
|
|
|
|
endDate: now
|
|
|
|
})
|
|
|
|
|
|
|
|
expect(result.groupInterval).to.equal('6 months')
|
|
|
|
expect(result.data).to.have.length.above(10).and.below(200)
|
|
|
|
})
|
|
|
|
|
|
|
|
it('Should automatically group by days', async function () {
|
|
|
|
const now = new Date()
|
|
|
|
const threeMonthsAgo = new Date()
|
|
|
|
threeMonthsAgo.setMonth(threeMonthsAgo.getMonth() - 3)
|
|
|
|
|
|
|
|
const result = await servers[0].videoStats.getTimeserieStats({
|
|
|
|
videoId: vodVideoId,
|
|
|
|
metric: 'aggregateWatchTime',
|
|
|
|
startDate: threeMonthsAgo,
|
|
|
|
endDate: now
|
|
|
|
})
|
|
|
|
|
|
|
|
expect(result.groupInterval).to.equal('2 days')
|
|
|
|
expect(result.data).to.have.length.above(10).and.below(200)
|
|
|
|
})
|
|
|
|
|
2022-04-07 10:53:35 +02:00
|
|
|
it('Should automatically group by hours', async function () {
|
|
|
|
const now = new Date()
|
|
|
|
const twoDaysAgo = new Date()
|
|
|
|
twoDaysAgo.setDate(twoDaysAgo.getDate() - 1)
|
|
|
|
|
|
|
|
const result = await servers[0].videoStats.getTimeserieStats({
|
|
|
|
videoId: vodVideoId,
|
|
|
|
metric: 'aggregateWatchTime',
|
|
|
|
startDate: twoDaysAgo,
|
|
|
|
endDate: now
|
|
|
|
})
|
|
|
|
|
2022-04-08 10:22:56 +02:00
|
|
|
expect(result.groupInterval).to.equal('1 hour')
|
2022-04-07 10:53:35 +02:00
|
|
|
expect(result.data).to.have.length.above(24).and.below(50)
|
|
|
|
|
|
|
|
expectInterval(result, 3600 * 1000)
|
|
|
|
expectTodayLastValue(result, 9)
|
|
|
|
})
|
|
|
|
|
|
|
|
it('Should automatically group by ten minutes', async function () {
|
|
|
|
const now = new Date()
|
|
|
|
const twoHoursAgo = new Date()
|
2022-04-08 10:22:56 +02:00
|
|
|
twoHoursAgo.setHours(twoHoursAgo.getHours() - 4)
|
2022-04-07 10:53:35 +02:00
|
|
|
|
|
|
|
const result = await servers[0].videoStats.getTimeserieStats({
|
|
|
|
videoId: vodVideoId,
|
|
|
|
metric: 'aggregateWatchTime',
|
|
|
|
startDate: twoHoursAgo,
|
|
|
|
endDate: now
|
|
|
|
})
|
|
|
|
|
2022-04-08 10:22:56 +02:00
|
|
|
expect(result.groupInterval).to.equal('10 minutes')
|
|
|
|
expect(result.data).to.have.length.above(20).and.below(30)
|
2022-04-07 10:53:35 +02:00
|
|
|
|
|
|
|
expectInterval(result, 60 * 10 * 1000)
|
2022-05-06 14:23:02 +02:00
|
|
|
expectTodayLastValue(result)
|
2022-04-07 10:53:35 +02:00
|
|
|
})
|
|
|
|
|
|
|
|
it('Should automatically group by one minute', async function () {
|
|
|
|
const now = new Date()
|
|
|
|
const thirtyAgo = new Date()
|
|
|
|
thirtyAgo.setMinutes(thirtyAgo.getMinutes() - 30)
|
|
|
|
|
|
|
|
const result = await servers[0].videoStats.getTimeserieStats({
|
|
|
|
videoId: vodVideoId,
|
|
|
|
metric: 'aggregateWatchTime',
|
|
|
|
startDate: thirtyAgo,
|
|
|
|
endDate: now
|
|
|
|
})
|
|
|
|
|
2022-04-08 10:22:56 +02:00
|
|
|
expect(result.groupInterval).to.equal('1 minute')
|
2022-04-07 10:53:35 +02:00
|
|
|
expect(result.data).to.have.length.above(20).and.below(40)
|
|
|
|
|
|
|
|
expectInterval(result, 60 * 1000)
|
2022-05-06 14:23:02 +02:00
|
|
|
expectTodayLastValue(result)
|
2022-04-07 10:53:35 +02:00
|
|
|
})
|
|
|
|
|
2022-03-24 13:36:47 +01:00
|
|
|
after(async function () {
|
|
|
|
await stopFfmpeg(command)
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
after(async function () {
|
|
|
|
await cleanupTests(servers)
|
|
|
|
})
|
|
|
|
})
|