]> git.immae.eu Git - github/Chocobozzz/PeerTube.git/blame - server/tests/api/videos/multiple-servers.ts
Fix video import tests
[github/Chocobozzz/PeerTube.git] / server / tests / api / videos / multiple-servers.ts
CommitLineData
a1587156 1/* eslint-disable @typescript-eslint/no-unused-expressions,@typescript-eslint/require-await */
0e1dc3e7 2
0e1dc3e7 3import * as chai from 'chai'
d50acfab 4import 'mocha'
b1a134ee 5import { join } from 'path'
f595d394 6import * as request from 'supertest'
b1f5b93e 7import { VideoPrivacy } from '../../../../shared/models/videos'
c5d31dba 8import { VideoComment, VideoCommentThreadTree } from '../../../../shared/models/videos/video-comment.model'
0e1dc3e7 9import {
b64c950a 10 addVideoChannel,
89231874 11 checkTmpIsEmpty,
7243f84d
C
12 checkVideoFilesWereRemoved,
13 cleanupTests,
b64c950a
C
14 completeVideoCheck,
15 createUser,
16 dateIsValid,
17 doubleFollow,
18 flushAndRunMultipleServers,
3cd0734f 19 getLocalVideos,
b64c950a
C
20 getVideo,
21 getVideoChannelsList,
22 getVideosList,
b64c950a
C
23 rateVideo,
24 removeVideo,
25 ServerInfo,
26 setAccessTokensToServers,
27 testImage,
28 updateVideo,
29 uploadVideo,
30 userLogin,
31 viewVideo,
32 wait,
33 webtorrentAdd
94565d52 34} from '../../../../shared/extra-utils'
c5d31dba 35import {
b64c950a
C
36 addVideoCommentReply,
37 addVideoCommentThread,
38 deleteVideoComment,
39 getVideoCommentThreads,
696d83fd
C
40 getVideoThreadComments,
41 findCommentId
94565d52
C
42} from '../../../../shared/extra-utils/videos/video-comments'
43import { waitJobs } from '../../../../shared/extra-utils/server/jobs'
0e1dc3e7
C
44
45const expect = chai.expect
46
9a27cdc2 47describe('Test multiple servers', function () {
0e1dc3e7
C
48 let servers: ServerInfo[] = []
49 const toRemove = []
50 let videoUUID = ''
5f04dd2f 51 let videoChannelId: number
0e1dc3e7
C
52
53 before(async function () {
54 this.timeout(120000)
55
56 servers = await flushAndRunMultipleServers(3)
57
58 // Get the access tokens
59 await setAccessTokensToServers(servers)
60
48dce1c9
C
61 {
62 const videoChannel = {
8a19bee1 63 name: 'super_channel_name',
08c1efbe 64 displayName: 'my channel',
48dce1c9
C
65 description: 'super channel'
66 }
a1587156
C
67 await addVideoChannel(servers[0].url, servers[0].accessToken, videoChannel)
68 const channelRes = await getVideoChannelsList(servers[0].url, 0, 1)
69 videoChannelId = channelRes.body.data[0].id
5f04dd2f 70 }
5f04dd2f 71
9a27cdc2
C
72 // Server 1 and server 2 follow each other
73 await doubleFollow(servers[0], servers[1])
74 // Server 1 and server 3 follow each other
75 await doubleFollow(servers[0], servers[2])
76 // Server 2 and server 3 follow each other
77 await doubleFollow(servers[1], servers[2])
0e1dc3e7
C
78 })
79
9a27cdc2 80 it('Should not have videos for all servers', async function () {
0e1dc3e7
C
81 for (const server of servers) {
82 const res = await getVideosList(server.url)
83 const videos = res.body.data
84 expect(videos).to.be.an('array')
85 expect(videos.length).to.equal(0)
86 }
87 })
88
9a27cdc2
C
89 describe('Should upload the video and propagate on each server', function () {
90 it('Should upload the video on server 1 and propagate on each server', async function () {
652b3056 91 this.timeout(25000)
0e1dc3e7
C
92
93 const videoAttributes = {
9a27cdc2 94 name: 'my super name for server 1',
0e1dc3e7
C
95 category: 5,
96 licence: 4,
9d3ef9fe 97 language: 'ja',
0e1dc3e7 98 nsfw: true,
9a27cdc2 99 description: 'my super description for server 1',
2422c46b 100 support: 'my super support text for server 1',
7519127b 101 originallyPublishedAt: '2019-02-10T13:38:14.449Z',
0e1dc3e7 102 tags: [ 'tag1p1', 'tag2p1' ],
5f04dd2f 103 channelId: videoChannelId,
0e1dc3e7
C
104 fixture: 'video_short1.webm'
105 }
106 await uploadVideo(servers[0].url, servers[0].accessToken, videoAttributes)
107
3cd0734f 108 await waitJobs(servers)
0e1dc3e7 109
9a27cdc2 110 // All servers should have this video
53a61317 111 let publishedAt: string = null
0e1dc3e7 112 for (const server of servers) {
48f07b4a 113 const isLocal = server.port === servers[0].port
b1f5b93e
C
114 const checkAttributes = {
115 name: 'my super name for server 1',
116 category: 5,
117 licence: 4,
9d3ef9fe 118 language: 'ja',
b1f5b93e
C
119 nsfw: true,
120 description: 'my super description for server 1',
2422c46b 121 support: 'my super support text for server 1',
7519127b 122 originallyPublishedAt: '2019-02-10T13:38:14.449Z',
b64c950a
C
123 account: {
124 name: 'root',
7243f84d 125 host: 'localhost:' + servers[0].port
b64c950a 126 },
b1f5b93e 127 isLocal,
53a61317 128 publishedAt,
b1f5b93e
C
129 duration: 10,
130 tags: [ 'tag1p1', 'tag2p1' ],
131 privacy: VideoPrivacy.PUBLIC,
47564bbe 132 commentsEnabled: true,
7f2cfe3a 133 downloadEnabled: true,
b1f5b93e 134 channel: {
f6eebcb3
C
135 displayName: 'my channel',
136 name: 'super_channel_name',
b1f5b93e
C
137 description: 'super channel',
138 isLocal
139 },
8b0d42ee 140 fixture: 'video_short1.webm',
b1f5b93e
C
141 files: [
142 {
143 resolution: 720,
144 size: 572456
145 }
146 ]
147 }
0e1dc3e7
C
148
149 const res = await getVideosList(server.url)
0e1dc3e7
C
150 const videos = res.body.data
151 expect(videos).to.be.an('array')
152 expect(videos.length).to.equal(1)
153 const video = videos[0]
0e1dc3e7 154
b1f5b93e 155 await completeVideoCheck(server.url, video, checkAttributes)
53a61317 156 publishedAt = video.publishedAt
0e1dc3e7
C
157 }
158 })
159
9a27cdc2 160 it('Should upload the video on server 2 and propagate on each server', async function () {
572f8d3d 161 this.timeout(50000)
0e1dc3e7 162
5f04dd2f
C
163 const user = {
164 username: 'user1',
165 password: 'super_password'
166 }
a1587156 167 await createUser({ url: servers[1].url, accessToken: servers[1].accessToken, username: user.username, password: user.password })
eec63bbc 168 const userAccessToken = await userLogin(servers[1], user)
5f04dd2f 169
0e1dc3e7 170 const videoAttributes = {
9a27cdc2 171 name: 'my super name for server 2',
0e1dc3e7
C
172 category: 4,
173 licence: 3,
9d3ef9fe 174 language: 'de',
0e1dc3e7 175 nsfw: true,
9a27cdc2 176 description: 'my super description for server 2',
2422c46b 177 support: 'my super support text for server 2',
0e1dc3e7 178 tags: [ 'tag1p2', 'tag2p2', 'tag3p2' ],
ac81d1a0
C
179 fixture: 'video_short2.webm',
180 thumbnailfile: 'thumbnail.jpg',
181 previewfile: 'preview.jpg'
0e1dc3e7 182 }
5f04dd2f 183 await uploadVideo(servers[1].url, userAccessToken, videoAttributes)
0e1dc3e7 184
572f8d3d 185 // Transcoding
3cd0734f 186 await waitJobs(servers)
0e1dc3e7 187
9a27cdc2 188 // All servers should have this video
0e1dc3e7 189 for (const server of servers) {
7243f84d 190 const isLocal = server.url === 'http://localhost:' + servers[1].port
b1f5b93e
C
191 const checkAttributes = {
192 name: 'my super name for server 2',
193 category: 4,
194 licence: 3,
9d3ef9fe 195 language: 'de',
b1f5b93e
C
196 nsfw: true,
197 description: 'my super description for server 2',
2422c46b 198 support: 'my super support text for server 2',
b64c950a
C
199 account: {
200 name: 'user1',
7243f84d 201 host: 'localhost:' + servers[1].port
b64c950a 202 },
b1f5b93e 203 isLocal,
47564bbe 204 commentsEnabled: true,
7f2cfe3a 205 downloadEnabled: true,
b1f5b93e
C
206 duration: 5,
207 tags: [ 'tag1p2', 'tag2p2', 'tag3p2' ],
208 privacy: VideoPrivacy.PUBLIC,
209 channel: {
f6eebcb3
C
210 displayName: 'Main user1 channel',
211 name: 'user1_channel',
b1f5b93e
C
212 description: 'super channel',
213 isLocal
214 },
215 fixture: 'video_short2.webm',
216 files: [
217 {
218 resolution: 240,
80b8ad2a 219 size: 189000
b1f5b93e
C
220 },
221 {
222 resolution: 360,
eed24d26 223 size: 278000
b1f5b93e
C
224 },
225 {
226 resolution: 480,
80b8ad2a 227 size: 384000
b1f5b93e
C
228 },
229 {
230 resolution: 720,
eed24d26 231 size: 706000
b1f5b93e 232 }
ac81d1a0
C
233 ],
234 thumbnailfile: 'thumbnail',
235 previewfile: 'preview'
b1f5b93e 236 }
0e1dc3e7
C
237
238 const res = await getVideosList(server.url)
0e1dc3e7
C
239 const videos = res.body.data
240 expect(videos).to.be.an('array')
241 expect(videos.length).to.equal(2)
242 const video = videos[1]
0e1dc3e7 243
b1f5b93e 244 await completeVideoCheck(server.url, video, checkAttributes)
0e1dc3e7
C
245 }
246 })
247
9a27cdc2 248 it('Should upload two videos on server 3 and propagate on each server', async function () {
0e1dc3e7
C
249 this.timeout(45000)
250
251 const videoAttributes1 = {
9a27cdc2 252 name: 'my super name for server 3',
0e1dc3e7
C
253 category: 6,
254 licence: 5,
9d3ef9fe 255 language: 'de',
0e1dc3e7 256 nsfw: true,
9a27cdc2 257 description: 'my super description for server 3',
2422c46b 258 support: 'my super support text for server 3',
0e1dc3e7
C
259 tags: [ 'tag1p3' ],
260 fixture: 'video_short3.webm'
261 }
262 await uploadVideo(servers[2].url, servers[2].accessToken, videoAttributes1)
263
264 const videoAttributes2 = {
9a27cdc2 265 name: 'my super name for server 3-2',
0e1dc3e7
C
266 category: 7,
267 licence: 6,
9d3ef9fe 268 language: 'ko',
0e1dc3e7 269 nsfw: false,
9a27cdc2 270 description: 'my super description for server 3-2',
2422c46b 271 support: 'my super support text for server 3-2',
0e1dc3e7
C
272 tags: [ 'tag2p3', 'tag3p3', 'tag4p3' ],
273 fixture: 'video_short.webm'
274 }
275 await uploadVideo(servers[2].url, servers[2].accessToken, videoAttributes2)
276
3cd0734f 277 await waitJobs(servers)
0e1dc3e7 278
9a27cdc2 279 // All servers should have this video
0e1dc3e7 280 for (const server of servers) {
7243f84d 281 const isLocal = server.url === 'http://localhost:' + servers[2].port
0e1dc3e7
C
282 const res = await getVideosList(server.url)
283
284 const videos = res.body.data
285 expect(videos).to.be.an('array')
286 expect(videos.length).to.equal(4)
287
288 // We not sure about the order of the two last uploads
289 let video1 = null
290 let video2 = null
9a27cdc2 291 if (videos[2].name === 'my super name for server 3') {
0e1dc3e7
C
292 video1 = videos[2]
293 video2 = videos[3]
294 } else {
295 video1 = videos[3]
296 video2 = videos[2]
297 }
298
b1f5b93e
C
299 const checkAttributesVideo1 = {
300 name: 'my super name for server 3',
301 category: 6,
302 licence: 5,
9d3ef9fe 303 language: 'de',
b1f5b93e
C
304 nsfw: true,
305 description: 'my super description for server 3',
2422c46b 306 support: 'my super support text for server 3',
b64c950a
C
307 account: {
308 name: 'root',
7243f84d 309 host: 'localhost:' + servers[2].port
b64c950a 310 },
b1f5b93e
C
311 isLocal,
312 duration: 5,
47564bbe 313 commentsEnabled: true,
7f2cfe3a 314 downloadEnabled: true,
b1f5b93e
C
315 tags: [ 'tag1p3' ],
316 privacy: VideoPrivacy.PUBLIC,
317 channel: {
f6eebcb3
C
318 displayName: 'Main root channel',
319 name: 'root_channel',
b1f5b93e
C
320 description: '',
321 isLocal
322 },
323 fixture: 'video_short3.webm',
324 files: [
325 {
326 resolution: 720,
327 size: 292677
328 }
329 ]
0e1dc3e7 330 }
b1f5b93e
C
331 await completeVideoCheck(server.url, video1, checkAttributesVideo1)
332
333 const checkAttributesVideo2 = {
334 name: 'my super name for server 3-2',
335 category: 7,
336 licence: 6,
9d3ef9fe 337 language: 'ko',
b1f5b93e
C
338 nsfw: false,
339 description: 'my super description for server 3-2',
2422c46b 340 support: 'my super support text for server 3-2',
b64c950a
C
341 account: {
342 name: 'root',
7243f84d 343 host: 'localhost:' + servers[2].port
b64c950a 344 },
47564bbe 345 commentsEnabled: true,
7f2cfe3a 346 downloadEnabled: true,
b1f5b93e
C
347 isLocal,
348 duration: 5,
349 tags: [ 'tag2p3', 'tag3p3', 'tag4p3' ],
350 privacy: VideoPrivacy.PUBLIC,
351 channel: {
f6eebcb3
C
352 displayName: 'Main root channel',
353 name: 'root_channel',
b1f5b93e
C
354 description: '',
355 isLocal
356 },
8b0d42ee 357 fixture: 'video_short.webm',
b1f5b93e
C
358 files: [
359 {
360 resolution: 720,
361 size: 218910
362 }
363 ]
0e1dc3e7 364 }
b1f5b93e 365 await completeVideoCheck(server.url, video2, checkAttributesVideo2)
0e1dc3e7
C
366 }
367 })
368 })
369
066e94c5
C
370 describe('It should list local videos', function () {
371 it('Should list only local videos on server 1', async function () {
372 const { body } = await getLocalVideos(servers[0].url)
373
374 expect(body.total).to.equal(1)
375 expect(body.data).to.be.an('array')
376 expect(body.data.length).to.equal(1)
377 expect(body.data[0].name).to.equal('my super name for server 1')
378 })
379
380 it('Should list only local videos on server 2', async function () {
381 const { body } = await getLocalVideos(servers[1].url)
382
383 expect(body.total).to.equal(1)
384 expect(body.data).to.be.an('array')
385 expect(body.data.length).to.equal(1)
386 expect(body.data[0].name).to.equal('my super name for server 2')
387 })
388
389 it('Should list only local videos on server 3', async function () {
390 const { body } = await getLocalVideos(servers[2].url)
391
392 expect(body.total).to.equal(2)
393 expect(body.data).to.be.an('array')
394 expect(body.data.length).to.equal(2)
395 expect(body.data[0].name).to.equal('my super name for server 3')
396 expect(body.data[1].name).to.equal('my super name for server 3-2')
397 })
398 })
399
0e1dc3e7 400 describe('Should seed the uploaded video', function () {
9a27cdc2 401 it('Should add the file 1 by asking server 3', async function () {
572f8d3d 402 this.timeout(10000)
0e1dc3e7
C
403
404 const res = await getVideosList(servers[2].url)
405
406 const video = res.body.data[0]
407 toRemove.push(res.body.data[2])
408 toRemove.push(res.body.data[3])
409
5f04dd2f
C
410 const res2 = await getVideo(servers[2].url, video.id)
411 const videoDetails = res2.body
412
b1f5b93e 413 const torrent = await webtorrentAdd(videoDetails.files[0].magnetUri, true)
0e1dc3e7
C
414 expect(torrent.files).to.be.an('array')
415 expect(torrent.files.length).to.equal(1)
416 expect(torrent.files[0].path).to.exist.and.to.not.equal('')
417 })
418
9a27cdc2 419 it('Should add the file 2 by asking server 1', async function () {
572f8d3d 420 this.timeout(10000)
0e1dc3e7
C
421
422 const res = await getVideosList(servers[0].url)
423
424 const video = res.body.data[1]
5f04dd2f
C
425 const res2 = await getVideo(servers[0].url, video.id)
426 const videoDetails = res2.body
0e1dc3e7 427
b1f5b93e 428 const torrent = await webtorrentAdd(videoDetails.files[0].magnetUri, true)
0e1dc3e7
C
429 expect(torrent.files).to.be.an('array')
430 expect(torrent.files.length).to.equal(1)
431 expect(torrent.files[0].path).to.exist.and.to.not.equal('')
432 })
433
9a27cdc2 434 it('Should add the file 3 by asking server 2', async function () {
572f8d3d 435 this.timeout(10000)
0e1dc3e7
C
436
437 const res = await getVideosList(servers[1].url)
438
439 const video = res.body.data[2]
5f04dd2f
C
440 const res2 = await getVideo(servers[1].url, video.id)
441 const videoDetails = res2.body
0e1dc3e7 442
b1f5b93e 443 const torrent = await webtorrentAdd(videoDetails.files[0].magnetUri, true)
0e1dc3e7
C
444 expect(torrent.files).to.be.an('array')
445 expect(torrent.files.length).to.equal(1)
446 expect(torrent.files[0].path).to.exist.and.to.not.equal('')
447 })
448
9a27cdc2 449 it('Should add the file 3-2 by asking server 1', async function () {
572f8d3d 450 this.timeout(10000)
0e1dc3e7
C
451
452 const res = await getVideosList(servers[0].url)
453
454 const video = res.body.data[3]
5f04dd2f
C
455 const res2 = await getVideo(servers[0].url, video.id)
456 const videoDetails = res2.body
0e1dc3e7 457
5f04dd2f 458 const torrent = await webtorrentAdd(videoDetails.files[0].magnetUri)
0e1dc3e7
C
459 expect(torrent.files).to.be.an('array')
460 expect(torrent.files.length).to.equal(1)
461 expect(torrent.files[0].path).to.exist.and.to.not.equal('')
462 })
14d3270f 463
9a27cdc2 464 it('Should add the file 2 in 360p by asking server 1', async function () {
572f8d3d 465 this.timeout(10000)
14d3270f
C
466
467 const res = await getVideosList(servers[0].url)
468
9a27cdc2 469 const video = res.body.data.find(v => v.name === 'my super name for server 2')
5f04dd2f
C
470 const res2 = await getVideo(servers[0].url, video.id)
471 const videoDetails = res2.body
472
5d00a3d7 473 const file = videoDetails.files.find(f => f.resolution.id === 360)
14d3270f
C
474 expect(file).not.to.be.undefined
475
476 const torrent = await webtorrentAdd(file.magnetUri)
477 expect(torrent.files).to.be.an('array')
478 expect(torrent.files.length).to.equal(1)
479 expect(torrent.files[0].path).to.exist.and.to.not.equal('')
480 })
0e1dc3e7
C
481 })
482
483 describe('Should update video views, likes and dislikes', function () {
9a27cdc2
C
484 let localVideosServer3 = []
485 let remoteVideosServer1 = []
486 let remoteVideosServer2 = []
487 let remoteVideosServer3 = []
0e1dc3e7
C
488
489 before(async function () {
490 const res1 = await getVideosList(servers[0].url)
9a27cdc2 491 remoteVideosServer1 = res1.body.data.filter(video => video.isLocal === false).map(video => video.uuid)
0e1dc3e7
C
492
493 const res2 = await getVideosList(servers[1].url)
9a27cdc2 494 remoteVideosServer2 = res2.body.data.filter(video => video.isLocal === false).map(video => video.uuid)
0e1dc3e7
C
495
496 const res3 = await getVideosList(servers[2].url)
9a27cdc2
C
497 localVideosServer3 = res3.body.data.filter(video => video.isLocal === true).map(video => video.uuid)
498 remoteVideosServer3 = res3.body.data.filter(video => video.isLocal === false).map(video => video.uuid)
0e1dc3e7
C
499 })
500
501 it('Should view multiple videos on owned servers', async function () {
6b616860 502 this.timeout(30000)
0e1dc3e7 503
b5c0e955 504 await viewVideo(servers[2].url, localVideosServer3[0])
511765c9 505 await wait(1000)
b5c0e955
C
506
507 await viewVideo(servers[2].url, localVideosServer3[0])
511765c9 508 await viewVideo(servers[2].url, localVideosServer3[1])
b5c0e955 509
511765c9 510 await wait(1000)
b5c0e955 511
0b353d1d
C
512 await viewVideo(servers[2].url, localVideosServer3[0])
513 await viewVideo(servers[2].url, localVideosServer3[0])
0e1dc3e7 514
3cd0734f 515 await waitJobs(servers)
0e1dc3e7 516
6b616860
C
517 // Wait the repeatable job
518 await wait(6000)
519
c883db6d
C
520 await waitJobs(servers)
521
0e1dc3e7
C
522 for (const server of servers) {
523 const res = await getVideosList(server.url)
524
525 const videos = res.body.data
9a27cdc2
C
526 const video0 = videos.find(v => v.uuid === localVideosServer3[0])
527 const video1 = videos.find(v => v.uuid === localVideosServer3[1])
5f04dd2f 528
1f3e9fec
C
529 expect(video0.views).to.equal(3)
530 expect(video1.views).to.equal(1)
0e1dc3e7
C
531 }
532 })
533
534 it('Should view multiple videos on each servers', async function () {
24163420 535 this.timeout(45000)
0e1dc3e7
C
536
537 const tasks: Promise<any>[] = []
1f3e9fec
C
538 tasks.push(viewVideo(servers[0].url, remoteVideosServer1[0]))
539 tasks.push(viewVideo(servers[1].url, remoteVideosServer2[0]))
540 tasks.push(viewVideo(servers[1].url, remoteVideosServer2[0]))
541 tasks.push(viewVideo(servers[2].url, remoteVideosServer3[0]))
542 tasks.push(viewVideo(servers[2].url, remoteVideosServer3[1]))
543 tasks.push(viewVideo(servers[2].url, remoteVideosServer3[1]))
544 tasks.push(viewVideo(servers[2].url, remoteVideosServer3[1]))
545 tasks.push(viewVideo(servers[2].url, localVideosServer3[1]))
546 tasks.push(viewVideo(servers[2].url, localVideosServer3[1]))
547 tasks.push(viewVideo(servers[2].url, localVideosServer3[1]))
0e1dc3e7
C
548
549 await Promise.all(tasks)
550
3cd0734f 551 await waitJobs(servers)
0e1dc3e7 552
6b616860 553 // Wait the repeatable job
24163420 554 await wait(16000)
6b616860 555
c883db6d
C
556 await waitJobs(servers)
557
0e1dc3e7
C
558 let baseVideos = null
559
560 for (const server of servers) {
561 const res = await getVideosList(server.url)
562
563 const videos = res.body.data
564
565 // Initialize base videos for future comparisons
566 if (baseVideos === null) {
567 baseVideos = videos
35a097b8 568 continue
0e1dc3e7
C
569 }
570
571 for (const baseVideo of baseVideos) {
572 const sameVideo = videos.find(video => video.name === baseVideo.name)
573 expect(baseVideo.views).to.equal(sameVideo.views)
574 }
575 }
576 })
577
578 it('Should like and dislikes videos on different services', async function () {
572f8d3d 579 this.timeout(20000)
0e1dc3e7 580
94a5ff8a 581 await rateVideo(servers[0].url, servers[0].accessToken, remoteVideosServer1[0], 'like')
2ba92871 582 await wait(500)
94a5ff8a 583 await rateVideo(servers[0].url, servers[0].accessToken, remoteVideosServer1[0], 'dislike')
2ba92871 584 await wait(500)
94a5ff8a
C
585 await rateVideo(servers[0].url, servers[0].accessToken, remoteVideosServer1[0], 'like')
586 await rateVideo(servers[2].url, servers[2].accessToken, localVideosServer3[1], 'like')
2ba92871 587 await wait(500)
94a5ff8a
C
588 await rateVideo(servers[2].url, servers[2].accessToken, localVideosServer3[1], 'dislike')
589 await rateVideo(servers[2].url, servers[2].accessToken, remoteVideosServer3[1], 'dislike')
2ba92871 590 await wait(500)
94a5ff8a 591 await rateVideo(servers[2].url, servers[2].accessToken, remoteVideosServer3[0], 'like')
0e1dc3e7 592
3cd0734f 593 await waitJobs(servers)
0e1dc3e7
C
594
595 let baseVideos = null
596 for (const server of servers) {
597 const res = await getVideosList(server.url)
598
599 const videos = res.body.data
600
601 // Initialize base videos for future comparisons
602 if (baseVideos === null) {
603 baseVideos = videos
35a097b8 604 continue
0e1dc3e7
C
605 }
606
35a097b8 607 for (const baseVideo of baseVideos) {
0e1dc3e7
C
608 const sameVideo = videos.find(video => video.name === baseVideo.name)
609 expect(baseVideo.likes).to.equal(sameVideo.likes)
610 expect(baseVideo.dislikes).to.equal(sameVideo.dislikes)
35a097b8 611 }
0e1dc3e7
C
612 }
613 })
614 })
615
616 describe('Should manipulate these videos', function () {
9a27cdc2 617 it('Should update the video 3 by asking server 3', async function () {
572f8d3d 618 this.timeout(10000)
0e1dc3e7
C
619
620 const attributes = {
621 name: 'my super video updated',
622 category: 10,
623 licence: 7,
9d3ef9fe 624 language: 'fr',
0e1dc3e7
C
625 nsfw: true,
626 description: 'my super description updated',
2422c46b 627 support: 'my super support text updated',
ac81d1a0
C
628 tags: [ 'tag_up_1', 'tag_up_2' ],
629 thumbnailfile: 'thumbnail.jpg',
7519127b 630 originallyPublishedAt: '2019-02-11T13:38:14.449Z',
ac81d1a0 631 previewfile: 'preview.jpg'
0e1dc3e7
C
632 }
633
634 await updateVideo(servers[2].url, servers[2].accessToken, toRemove[0].id, attributes)
635
3cd0734f 636 await waitJobs(servers)
0e1dc3e7
C
637 })
638
9a27cdc2 639 it('Should have the video 3 updated on each server', async function () {
572f8d3d 640 this.timeout(10000)
0e1dc3e7
C
641
642 for (const server of servers) {
643 const res = await getVideosList(server.url)
644
645 const videos = res.body.data
646 const videoUpdated = videos.find(video => video.name === 'my super video updated')
0e1dc3e7 647 expect(!!videoUpdated).to.be.true
0e1dc3e7 648
7243f84d 649 const isLocal = server.url === 'http://localhost:' + servers[2].port
b1f5b93e
C
650 const checkAttributes = {
651 name: 'my super video updated',
652 category: 10,
653 licence: 7,
9d3ef9fe 654 language: 'fr',
b1f5b93e
C
655 nsfw: true,
656 description: 'my super description updated',
2422c46b 657 support: 'my super support text updated',
7519127b 658 originallyPublishedAt: '2019-02-11T13:38:14.449Z',
b64c950a
C
659 account: {
660 name: 'root',
7243f84d 661 host: 'localhost:' + servers[2].port
b64c950a 662 },
b1f5b93e
C
663 isLocal,
664 duration: 5,
47564bbe 665 commentsEnabled: true,
7f2cfe3a 666 downloadEnabled: true,
b1f5b93e
C
667 tags: [ 'tag_up_1', 'tag_up_2' ],
668 privacy: VideoPrivacy.PUBLIC,
669 channel: {
f6eebcb3
C
670 displayName: 'Main root channel',
671 name: 'root_channel',
b1f5b93e
C
672 description: '',
673 isLocal
674 },
675 fixture: 'video_short3.webm',
676 files: [
677 {
678 resolution: 720,
679 size: 292677
680 }
ac81d1a0
C
681 ],
682 thumbnailfile: 'thumbnail',
683 previewfile: 'preview'
b1f5b93e
C
684 }
685 await completeVideoCheck(server.url, videoUpdated, checkAttributes)
0e1dc3e7
C
686 }
687 })
688
9a27cdc2 689 it('Should remove the videos 3 and 3-2 by asking server 3', async function () {
572f8d3d 690 this.timeout(10000)
0e1dc3e7
C
691
692 await removeVideo(servers[2].url, servers[2].accessToken, toRemove[0].id)
693 await removeVideo(servers[2].url, servers[2].accessToken, toRemove[1].id)
694
3cd0734f 695 await waitJobs(servers)
0e1dc3e7
C
696 })
697
f05a1c30
C
698 it('Should not have files of videos 3 and 3-2 on each server', async function () {
699 for (const server of servers) {
ffc65cbd
C
700 await checkVideoFilesWereRemoved(toRemove[0].uuid, server.internalServerNumber)
701 await checkVideoFilesWereRemoved(toRemove[1].uuid, server.internalServerNumber)
f05a1c30
C
702 }
703 })
704
9a27cdc2 705 it('Should have videos 1 and 3 on each server', async function () {
0e1dc3e7
C
706 for (const server of servers) {
707 const res = await getVideosList(server.url)
708
709 const videos = res.body.data
710 expect(videos).to.be.an('array')
711 expect(videos.length).to.equal(2)
712 expect(videos[0].name).not.to.equal(videos[1].name)
713 expect(videos[0].name).not.to.equal(toRemove[0].name)
714 expect(videos[1].name).not.to.equal(toRemove[0].name)
715 expect(videos[0].name).not.to.equal(toRemove[1].name)
716 expect(videos[1].name).not.to.equal(toRemove[1].name)
717
9a27cdc2 718 videoUUID = videos.find(video => video.name === 'my super name for server 1').uuid
0e1dc3e7
C
719 }
720 })
721
9a27cdc2 722 it('Should get the same video by UUID on each server', async function () {
0e1dc3e7
C
723 let baseVideo = null
724 for (const server of servers) {
725 const res = await getVideo(server.url, videoUUID)
726
727 const video = res.body
728
729 if (baseVideo === null) {
730 baseVideo = video
35a097b8 731 continue
0e1dc3e7
C
732 }
733
734 expect(baseVideo.name).to.equal(video.name)
735 expect(baseVideo.uuid).to.equal(video.uuid)
5d00a3d7
C
736 expect(baseVideo.category.id).to.equal(video.category.id)
737 expect(baseVideo.language.id).to.equal(video.language.id)
738 expect(baseVideo.licence.id).to.equal(video.licence.id)
0e1dc3e7 739 expect(baseVideo.nsfw).to.equal(video.nsfw)
b64c950a
C
740 expect(baseVideo.account.name).to.equal(video.account.name)
741 expect(baseVideo.account.displayName).to.equal(video.account.displayName)
742 expect(baseVideo.account.url).to.equal(video.account.url)
743 expect(baseVideo.account.host).to.equal(video.account.host)
0e1dc3e7
C
744 expect(baseVideo.tags).to.deep.equal(video.tags)
745 }
746 })
747
9a27cdc2 748 it('Should get the preview from each server', async function () {
0e1dc3e7
C
749 for (const server of servers) {
750 const res = await getVideo(server.url, videoUUID)
751 const video = res.body
752
7b0956ec 753 await testImage(server.url, 'video_short1-preview.webm', video.previewPath)
0e1dc3e7
C
754 }
755 })
756 })
757
d50acfab 758 describe('Should comment these videos', function () {
73c08093
C
759 let childOfFirstChild: VideoCommentThreadTree
760
d50acfab
C
761 it('Should add comment (threads and replies)', async function () {
762 this.timeout(25000)
763
764 {
765 const text = 'my super first comment'
a1587156 766 await addVideoCommentThread(servers[0].url, servers[0].accessToken, videoUUID, text)
d50acfab
C
767 }
768
769 {
770 const text = 'my super second comment'
a1587156 771 await addVideoCommentThread(servers[2].url, servers[2].accessToken, videoUUID, text)
d50acfab
C
772 }
773
3cd0734f 774 await waitJobs(servers)
d50acfab
C
775
776 {
696d83fd 777 const threadId = await findCommentId(servers[1].url, videoUUID, 'my super first comment')
d50acfab
C
778
779 const text = 'my super answer to thread 1'
a1587156 780 await addVideoCommentReply(servers[1].url, servers[1].accessToken, videoUUID, threadId, text)
d50acfab
C
781 }
782
3cd0734f 783 await waitJobs(servers)
d50acfab
C
784
785 {
696d83fd 786 const threadId = await findCommentId(servers[2].url, videoUUID, 'my super first comment')
d50acfab
C
787
788 const res2 = await getVideoThreadComments(servers[2].url, videoUUID, threadId)
789 const childCommentId = res2.body.children[0].comment.id
790
791 const text3 = 'my second answer to thread 1'
a1587156 792 await addVideoCommentReply(servers[2].url, servers[2].accessToken, videoUUID, threadId, text3)
d50acfab
C
793
794 const text2 = 'my super answer to answer of thread 1'
a1587156 795 await addVideoCommentReply(servers[2].url, servers[2].accessToken, videoUUID, childCommentId, text2)
d50acfab
C
796 }
797
3cd0734f 798 await waitJobs(servers)
d50acfab
C
799 })
800
801 it('Should have these threads', async function () {
802 for (const server of servers) {
803 const res = await getVideoCommentThreads(server.url, videoUUID, 0, 5)
804
805 expect(res.body.total).to.equal(2)
806 expect(res.body.data).to.be.an('array')
807 expect(res.body.data).to.have.lengthOf(2)
808
809 {
810 const comment: VideoComment = res.body.data.find(c => c.text === 'my super first comment')
811 expect(comment).to.not.be.undefined
812 expect(comment.inReplyToCommentId).to.be.null
813 expect(comment.account.name).to.equal('root')
7243f84d 814 expect(comment.account.host).to.equal('localhost:' + servers[0].port)
d50acfab
C
815 expect(comment.totalReplies).to.equal(3)
816 expect(dateIsValid(comment.createdAt as string)).to.be.true
817 expect(dateIsValid(comment.updatedAt as string)).to.be.true
818 }
819
820 {
821 const comment: VideoComment = res.body.data.find(c => c.text === 'my super second comment')
822 expect(comment).to.not.be.undefined
823 expect(comment.inReplyToCommentId).to.be.null
824 expect(comment.account.name).to.equal('root')
7243f84d 825 expect(comment.account.host).to.equal('localhost:' + servers[2].port)
d50acfab
C
826 expect(comment.totalReplies).to.equal(0)
827 expect(dateIsValid(comment.createdAt as string)).to.be.true
828 expect(dateIsValid(comment.updatedAt as string)).to.be.true
829 }
830 }
831 })
832
833 it('Should have these comments', async function () {
834 for (const server of servers) {
835 const res1 = await getVideoCommentThreads(server.url, videoUUID, 0, 5)
836 const threadId = res1.body.data.find(c => c.text === 'my super first comment').id
837
838 const res2 = await getVideoThreadComments(server.url, videoUUID, threadId)
839
840 const tree: VideoCommentThreadTree = res2.body
841 expect(tree.comment.text).equal('my super first comment')
842 expect(tree.comment.account.name).equal('root')
7243f84d 843 expect(tree.comment.account.host).equal('localhost:' + servers[0].port)
d50acfab
C
844 expect(tree.children).to.have.lengthOf(2)
845
846 const firstChild = tree.children[0]
847 expect(firstChild.comment.text).to.equal('my super answer to thread 1')
848 expect(firstChild.comment.account.name).equal('root')
7243f84d 849 expect(firstChild.comment.account.host).equal('localhost:' + servers[1].port)
d50acfab
C
850 expect(firstChild.children).to.have.lengthOf(1)
851
73c08093 852 childOfFirstChild = firstChild.children[0]
d50acfab
C
853 expect(childOfFirstChild.comment.text).to.equal('my super answer to answer of thread 1')
854 expect(childOfFirstChild.comment.account.name).equal('root')
7243f84d 855 expect(childOfFirstChild.comment.account.host).equal('localhost:' + servers[2].port)
d50acfab
C
856 expect(childOfFirstChild.children).to.have.lengthOf(0)
857
858 const secondChild = tree.children[1]
859 expect(secondChild.comment.text).to.equal('my second answer to thread 1')
860 expect(secondChild.comment.account.name).equal('root')
7243f84d 861 expect(secondChild.comment.account.host).equal('localhost:' + servers[2].port)
d50acfab
C
862 expect(secondChild.children).to.have.lengthOf(0)
863 }
864 })
47564bbe 865
73c08093
C
866 it('Should delete a reply', async function () {
867 this.timeout(10000)
868
869 await deleteVideoComment(servers[2].url, servers[2].accessToken, videoUUID, childOfFirstChild.comment.id)
870
3cd0734f 871 await waitJobs(servers)
73c08093
C
872 })
873
69222afa 874 it('Should have this comment marked as deleted', async function () {
73c08093
C
875 for (const server of servers) {
876 const res1 = await getVideoCommentThreads(server.url, videoUUID, 0, 5)
877 const threadId = res1.body.data.find(c => c.text === 'my super first comment').id
878
879 const res2 = await getVideoThreadComments(server.url, videoUUID, threadId)
880
881 const tree: VideoCommentThreadTree = res2.body
882 expect(tree.comment.text).equal('my super first comment')
883
884 const firstChild = tree.children[0]
885 expect(firstChild.comment.text).to.equal('my super answer to thread 1')
69222afa
JM
886 expect(firstChild.children).to.have.lengthOf(1)
887
888 const deletedComment = firstChild.children[0].comment
889 expect(deletedComment.isDeleted).to.be.true
890 expect(deletedComment.deletedAt).to.not.be.null
891 expect(deletedComment.account).to.be.null
892 expect(deletedComment.text).to.equal('')
73c08093
C
893
894 const secondChild = tree.children[1]
895 expect(secondChild.comment.text).to.equal('my second answer to thread 1')
896 }
897 })
898
4cb6d457
C
899 it('Should delete the thread comments', async function () {
900 this.timeout(10000)
901
a1587156 902 const res = await getVideoCommentThreads(servers[0].url, videoUUID, 0, 5)
511765c9 903 const threadId = res.body.data.find(c => c.text === 'my super first comment').id
a1587156 904 await deleteVideoComment(servers[0].url, servers[0].accessToken, videoUUID, threadId)
4cb6d457 905
3cd0734f 906 await waitJobs(servers)
4cb6d457
C
907 })
908
69222afa 909 it('Should have the threads marked as deleted on other servers too', async function () {
4cb6d457
C
910 for (const server of servers) {
911 const res = await getVideoCommentThreads(server.url, videoUUID, 0, 5)
912
69222afa 913 expect(res.body.total).to.equal(2)
4cb6d457 914 expect(res.body.data).to.be.an('array')
69222afa 915 expect(res.body.data).to.have.lengthOf(2)
4cb6d457
C
916
917 {
918 const comment: VideoComment = res.body.data[0]
919 expect(comment).to.not.be.undefined
920 expect(comment.inReplyToCommentId).to.be.null
921 expect(comment.account.name).to.equal('root')
7243f84d 922 expect(comment.account.host).to.equal('localhost:' + servers[2].port)
4cb6d457
C
923 expect(comment.totalReplies).to.equal(0)
924 expect(dateIsValid(comment.createdAt as string)).to.be.true
925 expect(dateIsValid(comment.updatedAt as string)).to.be.true
926 }
69222afa
JM
927
928 {
929 const deletedComment: VideoComment = res.body.data[1]
930 expect(deletedComment).to.not.be.undefined
931 expect(deletedComment.isDeleted).to.be.true
932 expect(deletedComment.deletedAt).to.not.be.null
933 expect(deletedComment.text).to.equal('')
934 expect(deletedComment.inReplyToCommentId).to.be.null
935 expect(deletedComment.account).to.be.null
936 expect(deletedComment.totalReplies).to.equal(3)
937 expect(dateIsValid(deletedComment.createdAt as string)).to.be.true
938 expect(dateIsValid(deletedComment.updatedAt as string)).to.be.true
939 expect(dateIsValid(deletedComment.deletedAt as string)).to.be.true
940 }
4cb6d457
C
941 }
942 })
943
511765c9 944 it('Should delete a remote thread by the origin server', async function () {
b5206dfc
JM
945 this.timeout(5000)
946
a1587156 947 const res = await getVideoCommentThreads(servers[0].url, videoUUID, 0, 5)
511765c9 948 const threadId = res.body.data.find(c => c.text === 'my super second comment').id
a1587156 949 await deleteVideoComment(servers[0].url, servers[0].accessToken, videoUUID, threadId)
511765c9
C
950
951 await waitJobs(servers)
952 })
953
69222afa 954 it('Should have the threads marked as deleted on other servers too', async function () {
511765c9
C
955 for (const server of servers) {
956 const res = await getVideoCommentThreads(server.url, videoUUID, 0, 5)
957
69222afa
JM
958 expect(res.body.total).to.equal(2)
959 expect(res.body.data).to.have.lengthOf(2)
960
961 {
962 const comment: VideoComment = res.body.data[0]
963 expect(comment.text).to.equal('')
964 expect(comment.isDeleted).to.be.true
965 expect(comment.createdAt).to.not.be.null
966 expect(comment.deletedAt).to.not.be.null
967 expect(comment.account).to.be.null
968 expect(comment.totalReplies).to.equal(0)
969 }
970
971 {
972 const comment: VideoComment = res.body.data[1]
973 expect(comment.text).to.equal('')
974 expect(comment.isDeleted).to.be.true
975 expect(comment.createdAt).to.not.be.null
976 expect(comment.deletedAt).to.not.be.null
977 expect(comment.account).to.be.null
978 expect(comment.totalReplies).to.equal(3)
979 }
511765c9
C
980 }
981 })
982
53a94c7c 983 it('Should disable comments and download', async function () {
47564bbe
C
984 this.timeout(20000)
985
986 const attributes = {
53a94c7c
C
987 commentsEnabled: false,
988 downloadEnabled: false
47564bbe
C
989 }
990
991 await updateVideo(servers[0].url, servers[0].accessToken, videoUUID, attributes)
992
3cd0734f 993 await waitJobs(servers)
47564bbe
C
994
995 for (const server of servers) {
996 const res = await getVideo(server.url, videoUUID)
997 expect(res.body.commentsEnabled).to.be.false
53a94c7c 998 expect(res.body.downloadEnabled).to.be.false
47564bbe
C
999
1000 const text = 'my super forbidden comment'
1001 await addVideoCommentThread(server.url, server.accessToken, videoUUID, text, 409)
1002 }
1003 })
d50acfab
C
1004 })
1005
f595d394
C
1006 describe('With minimum parameters', function () {
1007 it('Should upload and propagate the video', async function () {
2186386c 1008 this.timeout(60000)
f595d394
C
1009
1010 const path = '/api/v1/videos/upload'
1011
1012 const req = request(servers[1].url)
1013 .post(path)
1014 .set('Accept', 'application/json')
1015 .set('Authorization', 'Bearer ' + servers[1].accessToken)
1016 .field('name', 'minimum parameters')
1017 .field('privacy', '1')
f595d394
C
1018 .field('channelId', '1')
1019
99d10301 1020 const filePath = join(__dirname, '..', '..', 'fixtures', 'video_short.webm')
f595d394
C
1021
1022 await req.attach('videofile', filePath)
a1587156 1023 .expect(200)
f595d394 1024
3cd0734f 1025 await waitJobs(servers)
f595d394
C
1026
1027 for (const server of servers) {
1028 const res = await getVideosList(server.url)
1029 const video = res.body.data.find(v => v.name === 'minimum parameters')
1030
7243f84d 1031 const isLocal = server.url === 'http://localhost:' + servers[1].port
b1f5b93e
C
1032 const checkAttributes = {
1033 name: 'minimum parameters',
1034 category: null,
1035 licence: null,
1036 language: null,
1037 nsfw: false,
1038 description: null,
2422c46b 1039 support: null,
b64c950a
C
1040 account: {
1041 name: 'root',
7243f84d 1042 host: 'localhost:' + servers[1].port
b64c950a 1043 },
b1f5b93e
C
1044 isLocal,
1045 duration: 5,
210709a9 1046 commentsEnabled: true,
7519127b 1047 downloadEnabled: true,
a1587156 1048 tags: [],
b1f5b93e
C
1049 privacy: VideoPrivacy.PUBLIC,
1050 channel: {
f6eebcb3
C
1051 displayName: 'Main root channel',
1052 name: 'root_channel',
b1f5b93e
C
1053 description: '',
1054 isLocal
1055 },
1056 fixture: 'video_short.webm',
1057 files: [
1058 {
1059 resolution: 720,
cdf4cb9e 1060 size: 72000
b1f5b93e
C
1061 },
1062 {
1063 resolution: 480,
cdf4cb9e 1064 size: 45000
b1f5b93e
C
1065 },
1066 {
1067 resolution: 360,
cdf4cb9e 1068 size: 34600
b1f5b93e
C
1069 },
1070 {
1071 resolution: 240,
cdf4cb9e 1072 size: 24770
b1f5b93e
C
1073 }
1074 ]
1075 }
1076 await completeVideoCheck(server.url, video, checkAttributes)
f595d394
C
1077 }
1078 })
1079 })
1080
89231874
C
1081 describe('TMP directory', function () {
1082 it('Should have an empty tmp directory', async function () {
1083 for (const server of servers) {
1084 await checkTmpIsEmpty(server)
1085 }
1086 })
1087 })
1088
7c3b7976
C
1089 after(async function () {
1090 await cleanupTests(servers)
0e1dc3e7
C
1091 })
1092})