]> git.immae.eu Git - github/Chocobozzz/PeerTube.git/blob - server/lib/actor-image.ts
Add banner tests
[github/Chocobozzz/PeerTube.git] / server / lib / actor-image.ts
1 import 'multer'
2 import { queue } from 'async'
3 import * as LRUCache from 'lru-cache'
4 import { extname, join } from 'path'
5 import { v4 as uuidv4 } from 'uuid'
6 import { ActorImageType } from '@shared/models'
7 import { retryTransactionWrapper } from '../helpers/database-utils'
8 import { processImage } from '../helpers/image-utils'
9 import { downloadImage } from '../helpers/requests'
10 import { CONFIG } from '../initializers/config'
11 import { ACTOR_IMAGES_SIZE, LRU_CACHE, QUEUE_CONCURRENCY } from '../initializers/constants'
12 import { sequelizeTypescript } from '../initializers/database'
13 import { MAccountDefault, MChannelDefault } from '../types/models'
14 import { deleteActorImageInstance, updateActorImageInstance } from './activitypub/actor'
15 import { sendUpdateActor } from './activitypub/send'
16
17 async function updateLocalActorImageFile (
18 accountOrChannel: MAccountDefault | MChannelDefault,
19 imagePhysicalFile: Express.Multer.File,
20 type: ActorImageType
21 ) {
22 const imageSize = type === ActorImageType.AVATAR
23 ? ACTOR_IMAGES_SIZE.AVATARS
24 : ACTOR_IMAGES_SIZE.BANNERS
25
26 const extension = extname(imagePhysicalFile.filename)
27
28 const imageName = uuidv4() + extension
29 const destination = join(CONFIG.STORAGE.ACTOR_IMAGES, imageName)
30 await processImage(imagePhysicalFile.path, destination, imageSize)
31
32 return retryTransactionWrapper(() => {
33 return sequelizeTypescript.transaction(async t => {
34 const actorImageInfo = {
35 name: imageName,
36 fileUrl: null,
37 onDisk: true
38 }
39
40 const updatedActor = await updateActorImageInstance(accountOrChannel.Actor, type, actorImageInfo, t)
41 await updatedActor.save({ transaction: t })
42
43 await sendUpdateActor(accountOrChannel, t)
44
45 return type === ActorImageType.AVATAR
46 ? updatedActor.Avatar
47 : updatedActor.Banner
48 })
49 })
50 }
51
52 async function deleteLocalActorImageFile (accountOrChannel: MAccountDefault | MChannelDefault, type: ActorImageType) {
53 return retryTransactionWrapper(() => {
54 return sequelizeTypescript.transaction(async t => {
55 const updatedActor = await deleteActorImageInstance(accountOrChannel.Actor, type, t)
56 await updatedActor.save({ transaction: t })
57
58 await sendUpdateActor(accountOrChannel, t)
59
60 return updatedActor.Avatar
61 })
62 })
63 }
64
65 type DownloadImageQueueTask = { fileUrl: string, filename: string, type: ActorImageType }
66
67 const downloadImageQueue = queue<DownloadImageQueueTask, Error>((task, cb) => {
68 const size = task.type === ActorImageType.AVATAR
69 ? ACTOR_IMAGES_SIZE.AVATARS
70 : ACTOR_IMAGES_SIZE.BANNERS
71
72 downloadImage(task.fileUrl, CONFIG.STORAGE.ACTOR_IMAGES, task.filename, size)
73 .then(() => cb())
74 .catch(err => cb(err))
75 }, QUEUE_CONCURRENCY.ACTOR_PROCESS_IMAGE)
76
77 function pushActorImageProcessInQueue (task: DownloadImageQueueTask) {
78 return new Promise<void>((res, rej) => {
79 downloadImageQueue.push(task, err => {
80 if (err) return rej(err)
81
82 return res()
83 })
84 })
85 }
86
87 // Unsafe so could returns paths that does not exist anymore
88 const actorImagePathUnsafeCache = new LRUCache<string, string>({ max: LRU_CACHE.ACTOR_IMAGE_STATIC.MAX_SIZE })
89
90 export {
91 actorImagePathUnsafeCache,
92 updateLocalActorImageFile,
93 deleteLocalActorImageFile,
94 pushActorImageProcessInQueue
95 }