]> git.immae.eu Git - github/Chocobozzz/PeerTube.git/blame - server/tests/api/server/config.ts
Merge branch 'release/3.2.0' into develop
[github/Chocobozzz/PeerTube.git] / server / tests / api / server / config.ts
CommitLineData
a1587156 1/* eslint-disable @typescript-eslint/no-unused-expressions,@typescript-eslint/require-await */
0e1dc3e7
C
2
3import 'mocha'
4import * as chai from 'chai'
09cababd
C
5import { About } from '../../../../shared/models/server/about.model'
6import { CustomConfig } from '../../../../shared/models/server/custom-config.model'
0e1dc3e7 7import {
7c3b7976 8 cleanupTests,
9639bd17 9 deleteCustomConfig,
7c3b7976 10 flushAndRunServer,
9639bd17 11 getAbout,
5bcfd029
C
12 getConfig,
13 getCustomConfig,
a1587156 14 killallServers,
8155db66 15 makeGetRequest,
a1587156 16 parallelTests,
5bcfd029 17 registerUser,
a1587156
C
18 reRunServer,
19 ServerInfo,
5bcfd029 20 setAccessTokensToServers,
a1587156
C
21 updateCustomConfig,
22 uploadVideo
94565d52 23} from '../../../../shared/extra-utils'
14e2014a 24import { ServerConfig } from '../../../../shared/models'
f2eb23cd 25import { HttpStatusCode } from '../../../../shared/core-utils/miscs/http-error-codes'
0e1dc3e7 26
5bcfd029
C
27const expect = chai.expect
28
48f07b4a 29function checkInitialConfig (server: ServerInfo, data: CustomConfig) {
40e87e9e
C
30 expect(data.instance.name).to.equal('PeerTube')
31 expect(data.instance.shortDescription).to.equal(
482fa503 32 'PeerTube, an ActivityPub-federated video streaming platform using P2P directly in your web browser.'
40e87e9e
C
33 )
34 expect(data.instance.description).to.equal('Welcome to this PeerTube instance!')
ccc00cb2 35
40e87e9e 36 expect(data.instance.terms).to.equal('No terms for now.')
8ae03c37 37 expect(data.instance.creationReason).to.be.empty
ccc00cb2
C
38 expect(data.instance.codeOfConduct).to.be.empty
39 expect(data.instance.moderationInformation).to.be.empty
40 expect(data.instance.administrator).to.be.empty
41 expect(data.instance.maintenanceLifetime).to.be.empty
42 expect(data.instance.businessModel).to.be.empty
be04c6fd 43 expect(data.instance.hardwareInformation).to.be.empty
ccc00cb2
C
44
45 expect(data.instance.languages).to.have.lengthOf(0)
46 expect(data.instance.categories).to.have.lengthOf(0)
47
40e87e9e 48 expect(data.instance.defaultClientRoute).to.equal('/videos/trending')
f8802489 49 expect(data.instance.isNSFW).to.be.false
40e87e9e
C
50 expect(data.instance.defaultNSFWPolicy).to.equal('display')
51 expect(data.instance.customizations.css).to.be.empty
52 expect(data.instance.customizations.javascript).to.be.empty
a4101923 53
40e87e9e
C
54 expect(data.services.twitter.username).to.equal('@Chocobozzz')
55 expect(data.services.twitter.whitelisted).to.be.false
a4101923 56
40e87e9e
C
57 expect(data.cache.previews.size).to.equal(1)
58 expect(data.cache.captions.size).to.equal(1)
b3d5cb92 59 expect(data.cache.torrents.size).to.equal(1)
a4101923 60
40e87e9e
C
61 expect(data.signup.enabled).to.be.true
62 expect(data.signup.limit).to.equal(4)
d9eaee39 63 expect(data.signup.requiresEmailVerification).to.be.false
a4101923 64
48f07b4a 65 expect(data.admin.email).to.equal('admin' + server.internalServerNumber + '@example.com')
a4101923
C
66 expect(data.contactForm.enabled).to.be.true
67
40e87e9e 68 expect(data.user.videoQuota).to.equal(5242880)
9ee92651 69 expect(data.user.videoQuotaDaily).to.equal(-1)
c6c0fa6c 70
40e87e9e 71 expect(data.transcoding.enabled).to.be.false
14e2014a 72 expect(data.transcoding.allowAdditionalExtensions).to.be.false
536598cf 73 expect(data.transcoding.allowAudioFiles).to.be.false
40e87e9e 74 expect(data.transcoding.threads).to.equal(2)
9129b769 75 expect(data.transcoding.concurrency).to.equal(2)
1896bca0 76 expect(data.transcoding.profile).to.equal('default')
40e87e9e
C
77 expect(data.transcoding.resolutions['240p']).to.be.true
78 expect(data.transcoding.resolutions['360p']).to.be.true
79 expect(data.transcoding.resolutions['480p']).to.be.true
80 expect(data.transcoding.resolutions['720p']).to.be.true
81 expect(data.transcoding.resolutions['1080p']).to.be.true
b7085c71 82 expect(data.transcoding.resolutions['1440p']).to.be.true
db714ab4 83 expect(data.transcoding.resolutions['2160p']).to.be.true
d7a25329 84 expect(data.transcoding.webtorrent.enabled).to.be.true
09209296
C
85 expect(data.transcoding.hls.enabled).to.be.true
86
c6c0fa6c 87 expect(data.live.enabled).to.be.false
c655c9ef 88 expect(data.live.allowReplay).to.be.false
c9bc850e 89 expect(data.live.maxDuration).to.equal(-1)
a056ca48
C
90 expect(data.live.maxInstanceLives).to.equal(20)
91 expect(data.live.maxUserLives).to.equal(3)
c6c0fa6c
C
92 expect(data.live.transcoding.enabled).to.be.false
93 expect(data.live.transcoding.threads).to.equal(2)
1896bca0 94 expect(data.live.transcoding.profile).to.equal('default')
c6c0fa6c
C
95 expect(data.live.transcoding.resolutions['240p']).to.be.false
96 expect(data.live.transcoding.resolutions['360p']).to.be.false
97 expect(data.live.transcoding.resolutions['480p']).to.be.false
98 expect(data.live.transcoding.resolutions['720p']).to.be.false
99 expect(data.live.transcoding.resolutions['1080p']).to.be.false
b7085c71 100 expect(data.live.transcoding.resolutions['1440p']).to.be.false
c6c0fa6c
C
101 expect(data.live.transcoding.resolutions['2160p']).to.be.false
102
9129b769 103 expect(data.import.videos.concurrency).to.equal(2)
5d08a6a7 104 expect(data.import.videos.http.enabled).to.be.true
a84b8fa5 105 expect(data.import.videos.torrent.enabled).to.be.true
7ccddd7b 106 expect(data.autoBlacklist.videos.ofUsers.enabled).to.be.false
5b9c965d
C
107
108 expect(data.followers.instance.enabled).to.be.true
14893eb7 109 expect(data.followers.instance.manualApproval).to.be.false
8424c402
C
110
111 expect(data.followings.instance.autoFollowBack.enabled).to.be.false
112 expect(data.followings.instance.autoFollowIndex.enabled).to.be.false
c9215a1b 113 expect(data.followings.instance.autoFollowIndex.indexUrl).to.equal('')
72c33e71
C
114
115 expect(data.broadcastMessage.enabled).to.be.false
116 expect(data.broadcastMessage.level).to.equal('info')
117 expect(data.broadcastMessage.message).to.equal('')
118 expect(data.broadcastMessage.dismissable).to.be.false
40e87e9e
C
119}
120
121function checkUpdatedConfig (data: CustomConfig) {
122 expect(data.instance.name).to.equal('PeerTube updated')
123 expect(data.instance.shortDescription).to.equal('my short description')
124 expect(data.instance.description).to.equal('my super description')
ccc00cb2 125
40e87e9e 126 expect(data.instance.terms).to.equal('my super terms')
8ae03c37 127 expect(data.instance.creationReason).to.equal('my super creation reason')
ccc00cb2
C
128 expect(data.instance.codeOfConduct).to.equal('my super coc')
129 expect(data.instance.moderationInformation).to.equal('my super moderation information')
130 expect(data.instance.administrator).to.equal('Kuja')
131 expect(data.instance.maintenanceLifetime).to.equal('forever')
132 expect(data.instance.businessModel).to.equal('my super business model')
be04c6fd 133 expect(data.instance.hardwareInformation).to.equal('2vCore 3GB RAM')
ccc00cb2
C
134
135 expect(data.instance.languages).to.deep.equal([ 'en', 'es' ])
136 expect(data.instance.categories).to.deep.equal([ 1, 2 ])
137
40e87e9e 138 expect(data.instance.defaultClientRoute).to.equal('/videos/recently-added')
f8802489 139 expect(data.instance.isNSFW).to.be.true
40e87e9e
C
140 expect(data.instance.defaultNSFWPolicy).to.equal('blur')
141 expect(data.instance.customizations.javascript).to.equal('alert("coucou")')
142 expect(data.instance.customizations.css).to.equal('body { background-color: red; }')
a4101923 143
40e87e9e
C
144 expect(data.services.twitter.username).to.equal('@Kuja')
145 expect(data.services.twitter.whitelisted).to.be.true
a4101923 146
40e87e9e
C
147 expect(data.cache.previews.size).to.equal(2)
148 expect(data.cache.captions.size).to.equal(3)
b3d5cb92 149 expect(data.cache.torrents.size).to.equal(4)
a4101923 150
40e87e9e
C
151 expect(data.signup.enabled).to.be.false
152 expect(data.signup.limit).to.equal(5)
576ad67a 153 expect(data.signup.requiresEmailVerification).to.be.false
a4101923 154
48f07b4a
C
155 // We override admin email in parallel tests, so skip this exception
156 if (parallelTests() === false) {
157 expect(data.admin.email).to.equal('superadmin1@example.com')
158 }
159
a4101923
C
160 expect(data.contactForm.enabled).to.be.false
161
40e87e9e 162 expect(data.user.videoQuota).to.equal(5242881)
bee0abff 163 expect(data.user.videoQuotaDaily).to.equal(318742)
a4101923 164
40e87e9e
C
165 expect(data.transcoding.enabled).to.be.true
166 expect(data.transcoding.threads).to.equal(1)
9129b769 167 expect(data.transcoding.concurrency).to.equal(3)
14e2014a 168 expect(data.transcoding.allowAdditionalExtensions).to.be.true
536598cf 169 expect(data.transcoding.allowAudioFiles).to.be.true
1896bca0 170 expect(data.transcoding.profile).to.equal('vod_profile')
40e87e9e
C
171 expect(data.transcoding.resolutions['240p']).to.be.false
172 expect(data.transcoding.resolutions['360p']).to.be.true
173 expect(data.transcoding.resolutions['480p']).to.be.true
174 expect(data.transcoding.resolutions['720p']).to.be.false
175 expect(data.transcoding.resolutions['1080p']).to.be.false
db714ab4 176 expect(data.transcoding.resolutions['2160p']).to.be.false
09209296 177 expect(data.transcoding.hls.enabled).to.be.false
d7a25329 178 expect(data.transcoding.webtorrent.enabled).to.be.true
a4101923 179
c6c0fa6c 180 expect(data.live.enabled).to.be.true
c655c9ef 181 expect(data.live.allowReplay).to.be.true
fb719404 182 expect(data.live.maxDuration).to.equal(5000)
a056ca48
C
183 expect(data.live.maxInstanceLives).to.equal(-1)
184 expect(data.live.maxUserLives).to.equal(10)
c6c0fa6c
C
185 expect(data.live.transcoding.enabled).to.be.true
186 expect(data.live.transcoding.threads).to.equal(4)
1896bca0 187 expect(data.live.transcoding.profile).to.equal('live_profile')
c6c0fa6c
C
188 expect(data.live.transcoding.resolutions['240p']).to.be.true
189 expect(data.live.transcoding.resolutions['360p']).to.be.true
190 expect(data.live.transcoding.resolutions['480p']).to.be.true
191 expect(data.live.transcoding.resolutions['720p']).to.be.true
192 expect(data.live.transcoding.resolutions['1080p']).to.be.true
193 expect(data.live.transcoding.resolutions['2160p']).to.be.true
194
9129b769 195 expect(data.import.videos.concurrency).to.equal(4)
5d08a6a7 196 expect(data.import.videos.http.enabled).to.be.false
a84b8fa5 197 expect(data.import.videos.torrent.enabled).to.be.false
7ccddd7b 198 expect(data.autoBlacklist.videos.ofUsers.enabled).to.be.true
5b9c965d
C
199
200 expect(data.followers.instance.enabled).to.be.false
14893eb7 201 expect(data.followers.instance.manualApproval).to.be.true
8424c402
C
202
203 expect(data.followings.instance.autoFollowBack.enabled).to.be.true
204 expect(data.followings.instance.autoFollowIndex.enabled).to.be.true
205 expect(data.followings.instance.autoFollowIndex.indexUrl).to.equal('https://updated.example.com')
72c33e71
C
206
207 expect(data.broadcastMessage.enabled).to.be.true
208 expect(data.broadcastMessage.level).to.equal('error')
209 expect(data.broadcastMessage.message).to.equal('super bad message')
210 expect(data.broadcastMessage.dismissable).to.be.true
40e87e9e
C
211}
212
0e1dc3e7
C
213describe('Test config', function () {
214 let server = null
215
216 before(async function () {
e212f887 217 this.timeout(30000)
48f07b4a 218
210feb6c 219 server = await flushAndRunServer(1)
fd206f0b 220 await setAccessTokensToServers([ server ])
0e1dc3e7
C
221 })
222
223 it('Should have a correct config on a server with registration enabled', async function () {
224 const res = await getConfig(server.url)
14e2014a 225 const data: ServerConfig = res.body
0e1dc3e7
C
226
227 expect(data.signup.allowed).to.be.true
228 })
229
230 it('Should have a correct config on a server with registration enabled and a users limit', async function () {
652b3056
C
231 this.timeout(5000)
232
47e0652b
C
233 await Promise.all([
234 registerUser(server.url, 'user1', 'super password'),
235 registerUser(server.url, 'user2', 'super password'),
236 registerUser(server.url, 'user3', 'super password')
237 ])
0e1dc3e7
C
238
239 const res = await getConfig(server.url)
14e2014a 240 const data: ServerConfig = res.body
0e1dc3e7
C
241
242 expect(data.signup.allowed).to.be.false
243 })
244
14e2014a
C
245 it('Should have the correct video allowed extensions', async function () {
246 const res = await getConfig(server.url)
247 const data: ServerConfig = res.body
248
249 expect(data.video.file.extensions).to.have.lengthOf(3)
250 expect(data.video.file.extensions).to.contain('.mp4')
251 expect(data.video.file.extensions).to.contain('.webm')
252 expect(data.video.file.extensions).to.contain('.ogv')
a4101923 253
f2eb23cd
RK
254 await uploadVideo(server.url, server.accessToken, { fixture: 'video_short.mkv' }, HttpStatusCode.UNSUPPORTED_MEDIA_TYPE_415)
255 await uploadVideo(server.url, server.accessToken, { fixture: 'sample.ogg' }, HttpStatusCode.UNSUPPORTED_MEDIA_TYPE_415)
b345a804 256
a4101923 257 expect(data.contactForm.enabled).to.be.true
14e2014a
C
258 })
259
fd206f0b
C
260 it('Should get the customized configuration', async function () {
261 const res = await getCustomConfig(server.url, server.accessToken)
00b5556c 262 const data = res.body as CustomConfig
fd206f0b 263
48f07b4a 264 checkInitialConfig(server, data)
fd206f0b
C
265 })
266
267 it('Should update the customized configuration', async function () {
40e87e9e 268 const newCustomConfig: CustomConfig = {
66b16caf
C
269 instance: {
270 name: 'PeerTube updated',
2e3a0215 271 shortDescription: 'my short description',
66b16caf 272 description: 'my super description',
00b5556c 273 terms: 'my super terms',
ccc00cb2
C
274 codeOfConduct: 'my super coc',
275
8ae03c37 276 creationReason: 'my super creation reason',
ccc00cb2
C
277 moderationInformation: 'my super moderation information',
278 administrator: 'Kuja',
279 maintenanceLifetime: 'forever',
280 businessModel: 'my super business model',
be04c6fd 281 hardwareInformation: '2vCore 3GB RAM',
ccc00cb2
C
282
283 languages: [ 'en', 'es' ],
284 categories: [ 1, 2 ],
285
f8802489 286 isNSFW: true,
0883b324 287 defaultNSFWPolicy: 'blur' as 'blur',
3da68f0a
RK
288
289 defaultClientRoute: '/videos/recently-added',
3da68f0a 290
00b5556c
C
291 customizations: {
292 javascript: 'alert("coucou")',
293 css: 'body { background-color: red; }'
294 }
66b16caf 295 },
7cd4d2ba
C
296 theme: {
297 default: 'default'
298 },
8be1afa1
C
299 services: {
300 twitter: {
301 username: '@Kuja',
302 whitelisted: true
303 }
304 },
fd206f0b
C
305 cache: {
306 previews: {
307 size: 2
40e87e9e
C
308 },
309 captions: {
310 size: 3
b3d5cb92
C
311 },
312 torrents: {
313 size: 4
fd206f0b
C
314 }
315 },
316 signup: {
317 enabled: false,
d9eaee39 318 limit: 5,
576ad67a 319 requiresEmailVerification: false
fd206f0b
C
320 },
321 admin: {
322 email: 'superadmin1@example.com'
323 },
a4101923
C
324 contactForm: {
325 enabled: false
326 },
fd206f0b 327 user: {
bee0abff
FA
328 videoQuota: 5242881,
329 videoQuotaDaily: 318742
fd206f0b
C
330 },
331 transcoding: {
332 enabled: true,
14e2014a 333 allowAdditionalExtensions: true,
536598cf 334 allowAudioFiles: true,
fd206f0b 335 threads: 1,
9129b769 336 concurrency: 3,
1896bca0 337 profile: 'vod_profile',
fd206f0b 338 resolutions: {
5c7d6508 339 '0p': false,
fd206f0b
C
340 '240p': false,
341 '360p': true,
342 '480p': true,
343 '720p': false,
db714ab4 344 '1080p': false,
b7085c71 345 '1440p': false,
db714ab4 346 '2160p': false
09209296 347 },
d7a25329
C
348 webtorrent: {
349 enabled: true
350 },
09209296
C
351 hls: {
352 enabled: false
fd206f0b 353 }
5d08a6a7 354 },
c6c0fa6c
C
355 live: {
356 enabled: true,
c655c9ef 357 allowReplay: true,
fb719404 358 maxDuration: 5000,
a056ca48
C
359 maxInstanceLives: -1,
360 maxUserLives: 10,
c6c0fa6c
C
361 transcoding: {
362 enabled: true,
363 threads: 4,
1896bca0 364 profile: 'live_profile',
c6c0fa6c
C
365 resolutions: {
366 '240p': true,
367 '360p': true,
368 '480p': true,
369 '720p': true,
370 '1080p': true,
b7085c71 371 '1440p': true,
c6c0fa6c
C
372 '2160p': true
373 }
374 }
375 },
5d08a6a7
C
376 import: {
377 videos: {
9129b769 378 concurrency: 4,
5d08a6a7
C
379 http: {
380 enabled: false
a84b8fa5
C
381 },
382 torrent: {
383 enabled: false
5d08a6a7
C
384 }
385 }
7ccddd7b 386 },
ba5d4a84
RK
387 trending: {
388 videos: {
389 algorithms: {
3d4e112d 390 enabled: [ 'best', 'hot', 'most-viewed', 'most-liked' ],
ba5d4a84
RK
391 default: 'hot'
392 }
393 }
394 },
7ccddd7b
JM
395 autoBlacklist: {
396 videos: {
397 ofUsers: {
398 enabled: true
399 }
400 }
5b9c965d
C
401 },
402 followers: {
403 instance: {
14893eb7
C
404 enabled: false,
405 manualApproval: true
5b9c965d 406 }
8424c402
C
407 },
408 followings: {
409 instance: {
410 autoFollowBack: {
411 enabled: true
412 },
413 autoFollowIndex: {
414 enabled: true,
415 indexUrl: 'https://updated.example.com'
416 }
417 }
72c33e71
C
418 },
419 broadcastMessage: {
420 enabled: true,
421 level: 'error',
422 message: 'super bad message',
423 dismissable: true
5fb2e288
C
424 },
425 search: {
426 remoteUri: {
427 anonymous: true,
428 users: true
429 },
430 searchIndex: {
431 enabled: true,
432 url: 'https://search.joinpeertube.org',
433 disableLocalSearch: true,
434 isDefaultSearch: true
435 }
fd206f0b
C
436 }
437 }
438 await updateCustomConfig(server.url, server.accessToken, newCustomConfig)
439
440 const res = await getCustomConfig(server.url, server.accessToken)
441 const data = res.body
442
40e87e9e 443 checkUpdatedConfig(data)
fd206f0b
C
444 })
445
14e2014a 446 it('Should have the correct updated video allowed extensions', async function () {
30bc55c8
C
447 this.timeout(10000)
448
14e2014a
C
449 const res = await getConfig(server.url)
450 const data: ServerConfig = res.body
451
30bc55c8 452 expect(data.video.file.extensions).to.have.length.above(4)
14e2014a
C
453 expect(data.video.file.extensions).to.contain('.mp4')
454 expect(data.video.file.extensions).to.contain('.webm')
455 expect(data.video.file.extensions).to.contain('.ogv')
456 expect(data.video.file.extensions).to.contain('.flv')
fccbbc1a 457 expect(data.video.file.extensions).to.contain('.wmv')
14e2014a 458 expect(data.video.file.extensions).to.contain('.mkv')
b345a804
C
459 expect(data.video.file.extensions).to.contain('.mp3')
460 expect(data.video.file.extensions).to.contain('.ogg')
461 expect(data.video.file.extensions).to.contain('.flac')
462
f2eb23cd
RK
463 await uploadVideo(server.url, server.accessToken, { fixture: 'video_short.mkv' }, HttpStatusCode.OK_200)
464 await uploadVideo(server.url, server.accessToken, { fixture: 'sample.ogg' }, HttpStatusCode.OK_200)
14e2014a
C
465 })
466
fd206f0b 467 it('Should have the configuration updated after a restart', async function () {
23e27dd5
C
468 this.timeout(10000)
469
fd206f0b
C
470 killallServers([ server ])
471
472 await reRunServer(server)
473
474 const res = await getCustomConfig(server.url, server.accessToken)
475 const data = res.body
476
40e87e9e 477 checkUpdatedConfig(data)
fd206f0b
C
478 })
479
36f9424f
C
480 it('Should fetch the about information', async function () {
481 const res = await getAbout(server.url)
482 const data: About = res.body
483
484 expect(data.instance.name).to.equal('PeerTube updated')
2e3a0215 485 expect(data.instance.shortDescription).to.equal('my short description')
36f9424f
C
486 expect(data.instance.description).to.equal('my super description')
487 expect(data.instance.terms).to.equal('my super terms')
be04c6fd
C
488 expect(data.instance.codeOfConduct).to.equal('my super coc')
489
490 expect(data.instance.creationReason).to.equal('my super creation reason')
491 expect(data.instance.moderationInformation).to.equal('my super moderation information')
492 expect(data.instance.administrator).to.equal('Kuja')
493 expect(data.instance.maintenanceLifetime).to.equal('forever')
494 expect(data.instance.businessModel).to.equal('my super business model')
495 expect(data.instance.hardwareInformation).to.equal('2vCore 3GB RAM')
496
497 expect(data.instance.languages).to.deep.equal([ 'en', 'es' ])
498 expect(data.instance.categories).to.deep.equal([ 1, 2 ])
36f9424f
C
499 })
500
fd206f0b 501 it('Should remove the custom configuration', async function () {
23e27dd5
C
502 this.timeout(10000)
503
fd206f0b
C
504 await deleteCustomConfig(server.url, server.accessToken)
505
506 const res = await getCustomConfig(server.url, server.accessToken)
507 const data = res.body
508
48f07b4a 509 checkInitialConfig(server, data)
fd206f0b
C
510 })
511
8155db66
C
512 it('Should enable frameguard', async function () {
513 this.timeout(25000)
514
515 {
516 const res = await makeGetRequest({
517 url: server.url,
518 path: '/api/v1/config',
519 statusCodeExpected: 200
520 })
521
522 expect(res.headers['x-frame-options']).to.exist
523 }
524
525 killallServers([ server ])
526
527 const config = {
528 security: {
529 frameguard: { enabled: false }
530 }
531 }
532 server = await reRunServer(server, config)
533
534 {
535 const res = await makeGetRequest({
536 url: server.url,
537 path: '/api/v1/config',
538 statusCodeExpected: 200
539 })
540
541 expect(res.headers['x-frame-options']).to.not.exist
542 }
543 })
544
7c3b7976
C
545 after(async function () {
546 await cleanupTests([ server ])
0e1dc3e7
C
547 })
548})