]> git.immae.eu Git - github/Chocobozzz/PeerTube.git/blame - server/tests/api/videos/video-imports.ts
Introduce import command
[github/Chocobozzz/PeerTube.git] / server / tests / api / videos / video-imports.ts
CommitLineData
a1587156 1/* eslint-disable @typescript-eslint/no-unused-expressions,@typescript-eslint/require-await */
590fb506 2
590fb506 3import 'mocha'
b488ba1e 4import * as chai from 'chai'
590fb506 5import {
6910f20f 6 areHttpImportTestsDisabled,
7c3b7976 7 cleanupTests,
590fb506
C
8 doubleFollow,
9 flushAndRunMultipleServers,
10 getMyUserInformation,
11 getMyVideos,
12 getVideo,
13 getVideosList,
3e17515e 14 immutableAssign,
6910f20f 15 ImportsCommand,
590fb506 16 ServerInfo,
b488ba1e 17 setAccessTokensToServers,
6910f20f
C
18 testCaptionFile,
19 testImage,
20 waitJobs
21} from '@shared/extra-utils'
22import { VideoDetails, VideoPrivacy, VideoResolution } from '@shared/models'
590fb506
C
23
24const expect = chai.expect
25
26describe('Test video imports', function () {
27 let servers: ServerInfo[] = []
28 let channelIdServer1: number
29 let channelIdServer2: number
30
b488ba1e
C
31 if (areHttpImportTestsDisabled()) return
32
a2470c9f
C
33 async function checkVideosServer1 (server: ServerInfo, idHttp: string, idMagnet: string, idTorrent: string) {
34 const resHttp = await getVideo(server.url, idHttp)
3e17515e
C
35 const videoHttp: VideoDetails = resHttp.body
36
37 expect(videoHttp.name).to.equal('small video - youtube')
f9b6d51f
C
38 // FIXME: youtube-dl seems broken
39 // expect(videoHttp.category.label).to.equal('News & Politics')
1dee8d68 40 // expect(videoHttp.licence.label).to.equal('Attribution')
3e17515e
C
41 expect(videoHttp.language.label).to.equal('Unknown')
42 expect(videoHttp.nsfw).to.be.false
43 expect(videoHttp.description).to.equal('this is a super description')
44 expect(videoHttp.tags).to.deep.equal([ 'tag1', 'tag2' ])
45 expect(videoHttp.files).to.have.lengthOf(1)
17036be5
C
46
47 const originallyPublishedAt = new Date(videoHttp.originallyPublishedAt)
48 expect(originallyPublishedAt.getDate()).to.equal(14)
49 expect(originallyPublishedAt.getMonth()).to.equal(0)
50 expect(originallyPublishedAt.getFullYear()).to.equal(2019)
3e17515e 51
a2470c9f 52 const resMagnet = await getVideo(server.url, idMagnet)
3e17515e 53 const videoMagnet: VideoDetails = resMagnet.body
a2470c9f 54 const resTorrent = await getVideo(server.url, idTorrent)
3e17515e
C
55 const videoTorrent: VideoDetails = resTorrent.body
56
57 for (const video of [ videoMagnet, videoTorrent ]) {
58 expect(video.category.label).to.equal('Misc')
59 expect(video.licence.label).to.equal('Unknown')
60 expect(video.language.label).to.equal('Unknown')
61 expect(video.nsfw).to.be.false
62 expect(video.description).to.equal('this is a super torrent description')
63 expect(video.tags).to.deep.equal([ 'tag_torrent1', 'tag_torrent2' ])
64 expect(video.files).to.have.lengthOf(1)
65 }
590fb506 66
3e17515e
C
67 expect(videoTorrent.name).to.contain('你好 世界 720p.mp4')
68 expect(videoMagnet.name).to.contain('super peertube2 video')
652c6416 69
a2470c9f
C
70 const bodyCaptions = await server.captionsCommand.listVideoCaptions({ videoId: idHttp })
71 expect(bodyCaptions.total).to.equal(2)
590fb506
C
72 }
73
a2470c9f
C
74 async function checkVideoServer2 (server: ServerInfo, id: number | string) {
75 const res = await getVideo(server.url, id)
652c6416 76 const video: VideoDetails = res.body
590fb506
C
77
78 expect(video.name).to.equal('my super name')
79 expect(video.category.label).to.equal('Entertainment')
80 expect(video.licence.label).to.equal('Public Domain Dedication')
81 expect(video.language.label).to.equal('English')
82 expect(video.nsfw).to.be.false
83 expect(video.description).to.equal('my super description')
84 expect(video.tags).to.deep.equal([ 'supertag1', 'supertag2' ])
85
86 expect(video.files).to.have.lengthOf(1)
652c6416 87
a2470c9f
C
88 const bodyCaptions = await server.captionsCommand.listVideoCaptions({ videoId: id })
89 expect(bodyCaptions.total).to.equal(2)
590fb506
C
90 }
91
92 before(async function () {
454c20fa 93 this.timeout(30_000)
590fb506
C
94
95 // Run servers
96 servers = await flushAndRunMultipleServers(2)
97
98 await setAccessTokensToServers(servers)
99
100 {
101 const res = await getMyUserInformation(servers[0].url, servers[0].accessToken)
a1587156 102 channelIdServer1 = res.body.videoChannels[0].id
590fb506
C
103 }
104
105 {
106 const res = await getMyUserInformation(servers[1].url, servers[1].accessToken)
a1587156 107 channelIdServer2 = res.body.videoChannels[0].id
590fb506
C
108 }
109
110 await doubleFollow(servers[0], servers[1])
111 })
112
3e17515e 113 it('Should import videos on server 1', async function () {
454c20fa 114 this.timeout(60_000)
590fb506 115
3e17515e 116 const baseAttributes = {
590fb506
C
117 channelId: channelIdServer1,
118 privacy: VideoPrivacy.PUBLIC
119 }
3e17515e
C
120
121 {
6910f20f
C
122 const attributes = immutableAssign(baseAttributes, { targetUrl: ImportsCommand.getYoutubeVideoUrl() })
123 const { video } = await servers[0].importsCommand.importVideo({ attributes })
124 expect(video.name).to.equal('small video - youtube')
53c06121 125
6910f20f
C
126 expect(video.thumbnailPath).to.match(new RegExp(`^/static/thumbnails/.+.jpg$`))
127 expect(video.previewPath).to.match(new RegExp(`^/lazy-static/previews/.+.jpg$`))
53c06121 128
6910f20f
C
129 await testImage(servers[0].url, 'video_import_thumbnail', video.thumbnailPath)
130 await testImage(servers[0].url, 'video_import_preview', video.previewPath)
ba6e9e8f 131
6910f20f 132 const bodyCaptions = await servers[0].captionsCommand.listVideoCaptions({ videoId: video.id })
a2470c9f 133 const videoCaptions = bodyCaptions.data
ba6e9e8f 134 expect(videoCaptions).to.have.lengthOf(2)
135
652c6416
C
136 const enCaption = videoCaptions.find(caption => caption.language.id === 'en')
137 expect(enCaption).to.exist
138 expect(enCaption.language.label).to.equal('English')
53c06121 139 expect(enCaption.captionPath).to.match(new RegExp(`^/lazy-static/video-captions/.+-en.vtt$`))
ba6e9e8f 140 await testCaptionFile(servers[0].url, enCaption.captionPath, `WEBVTT
652c6416
C
141Kind: captions
142Language: en
ba6e9e8f 143
652c6416
C
14400:00:01.600 --> 00:00:04.200
145English (US)
ba6e9e8f 146
652c6416
C
14700:00:05.900 --> 00:00:07.999
148This is a subtitle in American English
ba6e9e8f 149
652c6416
C
15000:00:10.000 --> 00:00:14.000
151Adding subtitles is very easy to do`)
ba6e9e8f 152
652c6416
C
153 const frCaption = videoCaptions.find(caption => caption.language.id === 'fr')
154 expect(frCaption).to.exist
155 expect(frCaption.language.label).to.equal('French')
53c06121 156 expect(frCaption.captionPath).to.match(new RegExp(`^/lazy-static/video-captions/.+-fr.vtt`))
ba6e9e8f 157 await testCaptionFile(servers[0].url, frCaption.captionPath, `WEBVTT
652c6416
C
158Kind: captions
159Language: fr
ba6e9e8f 160
652c6416
C
16100:00:01.600 --> 00:00:04.200
162Français (FR)
ba6e9e8f 163
652c6416
C
16400:00:05.900 --> 00:00:07.999
165C'est un sous-titre français
ba6e9e8f 166
652c6416
C
16700:00:10.000 --> 00:00:14.000
168Ajouter un sous-titre est vraiment facile`)
3e17515e
C
169 }
170
171 {
172 const attributes = immutableAssign(baseAttributes, {
6910f20f 173 magnetUri: ImportsCommand.getMagnetURI(),
3e17515e
C
174 description: 'this is a super torrent description',
175 tags: [ 'tag_torrent1', 'tag_torrent2' ]
176 })
6910f20f
C
177 const { video } = await servers[0].importsCommand.importVideo({ attributes })
178 expect(video.name).to.equal('super peertube2 video')
3e17515e
C
179 }
180
181 {
182 const attributes = immutableAssign(baseAttributes, {
9a7fd960 183 torrentfile: 'video-720p.torrent' as any,
3e17515e
C
184 description: 'this is a super torrent description',
185 tags: [ 'tag_torrent1', 'tag_torrent2' ]
186 })
6910f20f
C
187 const { video } = await servers[0].importsCommand.importVideo({ attributes })
188 expect(video.name).to.equal('你好 世界 720p.mp4')
3e17515e 189 }
590fb506
C
190 })
191
3e17515e
C
192 it('Should list the videos to import in my videos on server 1', async function () {
193 const res = await getMyVideos(servers[0].url, servers[0].accessToken, 0, 5, 'createdAt')
590fb506 194
3e17515e 195 expect(res.body.total).to.equal(3)
590fb506
C
196
197 const videos = res.body.data
3e17515e 198 expect(videos).to.have.lengthOf(3)
590fb506 199 expect(videos[0].name).to.equal('small video - youtube')
3e17515e
C
200 expect(videos[1].name).to.equal('super peertube2 video')
201 expect(videos[2].name).to.equal('你好 世界 720p.mp4')
590fb506
C
202 })
203
3e17515e 204 it('Should list the videos to import in my imports on server 1', async function () {
6910f20f
C
205 const { total, data: videoImports } = await servers[0].importsCommand.getMyVideoImports({ sort: '-createdAt' })
206 expect(total).to.equal(3)
3e17515e 207
3e17515e
C
208 expect(videoImports).to.have.lengthOf(3)
209
6910f20f 210 expect(videoImports[2].targetUrl).to.equal(ImportsCommand.getYoutubeVideoUrl())
3e17515e
C
211 expect(videoImports[2].magnetUri).to.be.null
212 expect(videoImports[2].torrentName).to.be.null
213 expect(videoImports[2].video.name).to.equal('small video - youtube')
590fb506 214
3e17515e 215 expect(videoImports[1].targetUrl).to.be.null
6910f20f 216 expect(videoImports[1].magnetUri).to.equal(ImportsCommand.getMagnetURI())
3e17515e
C
217 expect(videoImports[1].torrentName).to.be.null
218 expect(videoImports[1].video.name).to.equal('super peertube2 video')
590fb506 219
3e17515e
C
220 expect(videoImports[0].targetUrl).to.be.null
221 expect(videoImports[0].magnetUri).to.be.null
222 expect(videoImports[0].torrentName).to.equal('video-720p.torrent')
223 expect(videoImports[0].video.name).to.equal('你好 世界 720p.mp4')
590fb506
C
224 })
225
3e17515e 226 it('Should have the video listed on the two instances', async function () {
454c20fa 227 this.timeout(120_000)
590fb506
C
228
229 await waitJobs(servers)
230
231 for (const server of servers) {
232 const res = await getVideosList(server.url)
3e17515e
C
233 expect(res.body.total).to.equal(3)
234 expect(res.body.data).to.have.lengthOf(3)
590fb506 235
3e17515e 236 const [ videoHttp, videoMagnet, videoTorrent ] = res.body.data
a2470c9f 237 await checkVideosServer1(server, videoHttp.uuid, videoMagnet.uuid, videoTorrent.uuid)
590fb506
C
238 }
239 })
240
241 it('Should import a video on server 2 with some fields', async function () {
454c20fa 242 this.timeout(60_000)
590fb506
C
243
244 const attributes = {
6910f20f 245 targetUrl: ImportsCommand.getYoutubeVideoUrl(),
3e17515e 246 channelId: channelIdServer2,
590fb506
C
247 privacy: VideoPrivacy.PUBLIC,
248 category: 10,
249 licence: 7,
250 language: 'en',
251 name: 'my super name',
252 description: 'my super description',
253 tags: [ 'supertag1', 'supertag2' ]
254 }
6910f20f
C
255 const { video } = await servers[1].importsCommand.importVideo({ attributes })
256 expect(video.name).to.equal('my super name')
590fb506
C
257 })
258
3e17515e 259 it('Should have the videos listed on the two instances', async function () {
454c20fa 260 this.timeout(120_000)
590fb506
C
261
262 await waitJobs(servers)
263
264 for (const server of servers) {
265 const res = await getVideosList(server.url)
3e17515e
C
266 expect(res.body.total).to.equal(4)
267 expect(res.body.data).to.have.lengthOf(4)
590fb506 268
a2470c9f 269 await checkVideoServer2(server, res.body.data[0].uuid)
3e17515e 270
a1587156 271 const [ , videoHttp, videoMagnet, videoTorrent ] = res.body.data
a2470c9f 272 await checkVideosServer1(server, videoHttp.uuid, videoMagnet.uuid, videoTorrent.uuid)
3e17515e
C
273 }
274 })
275
276 it('Should import a video that will be transcoded', async function () {
454c20fa 277 this.timeout(120_000)
3e17515e
C
278
279 const attributes = {
280 name: 'transcoded video',
6910f20f 281 magnetUri: ImportsCommand.getMagnetURI(),
3e17515e
C
282 channelId: channelIdServer2,
283 privacy: VideoPrivacy.PUBLIC
284 }
6910f20f
C
285 const { video } = await servers[1].importsCommand.importVideo({ attributes })
286 const videoUUID = video.uuid
3e17515e
C
287
288 await waitJobs(servers)
289
290 for (const server of servers) {
291 const res = await getVideo(server.url, videoUUID)
292 const video: VideoDetails = res.body
293
294 expect(video.name).to.equal('transcoded video')
295 expect(video.files).to.have.lengthOf(4)
590fb506
C
296 }
297 })
298
454c20fa
RK
299 it('Should import no HDR version on a HDR video', async function () {
300 this.timeout(120_000)
301
302 const config = {
303 transcoding: {
304 enabled: true,
305 resolutions: {
306 '240p': false,
307 '360p': false,
308 '480p': false,
309 '720p': false,
310 '1080p': true, // the resulting resolution shouldn't be higher than this, and not vp9.2/av01
311 '1440p': false,
312 '2160p': false
313 },
314 webtorrent: { enabled: true },
315 hls: { enabled: false }
316 },
317 import: {
318 videos: {
319 http: {
320 enabled: true
321 },
322 torrent: {
323 enabled: true
324 }
325 }
326 }
327 }
65e6e260 328 await servers[0].configCommand.updateCustomSubConfig({ newConfig: config })
454c20fa
RK
329
330 const attributes = {
331 name: 'hdr video',
6910f20f 332 targetUrl: ImportsCommand.getYoutubeHDRVideoUrl(),
454c20fa
RK
333 channelId: channelIdServer1,
334 privacy: VideoPrivacy.PUBLIC
335 }
6910f20f
C
336 const { video: videoImported } = await servers[0].importsCommand.importVideo({ attributes })
337 const videoUUID = videoImported.uuid
454c20fa
RK
338
339 await waitJobs(servers)
340
341 // test resolution
342 const res2 = await getVideo(servers[0].url, videoUUID)
343 const video: VideoDetails = res2.body
344 expect(video.name).to.equal('hdr video')
345 const maxResolution = Math.max.apply(Math, video.files.map(function (o) { return o.resolution.id }))
346 expect(maxResolution, 'expected max resolution not met').to.equals(VideoResolution.H_1080P)
347 })
348
7c3b7976
C
349 after(async function () {
350 await cleanupTests(servers)
590fb506
C
351 })
352})