PeerTube/server/core/lib/local-actor.ts

102 lines
3.7 KiB
TypeScript
Raw Normal View History

import { remove } from 'fs-extra/esm'
import { join } from 'path'
import { Transaction } from 'sequelize'
import { ActivityPubActorType, ActorImageType, ActorImageType_Type } from '@peertube/peertube-models'
import { ActorModel } from '@server/models/actor/actor.js'
import { buildUUID, getLowercaseExtension } from '@peertube/peertube-node-utils'
import { retryTransactionWrapper } from '../helpers/database-utils.js'
import { CONFIG } from '../initializers/config.js'
import { ACTOR_IMAGES_SIZE, WEBSERVER } from '../initializers/constants.js'
import { sequelizeTypescript } from '../initializers/database.js'
import { MAccountDefault, MActor, MChannelDefault } from '../types/models/index.js'
import { deleteActorImages, updateActorImages } from './activitypub/actors/index.js'
import { sendUpdateActor } from './activitypub/send/index.js'
import { processImageFromWorker } from './worker/parent-process.js'
2023-06-06 15:59:51 +02:00
export function buildActorInstance (type: ActivityPubActorType, url: string, preferredUsername: string) {
2021-06-03 16:02:29 +02:00
return new ActorModel({
type,
url,
preferredUsername,
publicKey: null,
privateKey: null,
followersCount: 0,
followingCount: 0,
inboxUrl: url + '/inbox',
outboxUrl: url + '/outbox',
sharedInboxUrl: WEBSERVER.URL + '/inbox',
followersUrl: url + '/followers',
followingUrl: url + '/following'
}) as MActor
}
2023-06-06 15:59:51 +02:00
export async function updateLocalActorImageFiles (
accountOrChannel: MAccountDefault | MChannelDefault,
2021-04-06 17:01:35 +02:00
imagePhysicalFile: Express.Multer.File,
type: ActorImageType_Type
2019-08-15 11:53:26 +02:00
) {
const processImageSize = async (imageSize: { width: number, height: number }) => {
const extension = getLowercaseExtension(imagePhysicalFile.filename)
const imageName = buildUUID() + extension
2023-07-11 11:23:51 +02:00
const destination = join(CONFIG.STORAGE.ACTOR_IMAGES_DIR, imageName)
2022-06-27 11:53:12 +02:00
await processImageFromWorker({ path: imagePhysicalFile.path, destination, newSize: imageSize, keepOriginal: true })
return {
imageName,
imageSize
}
}
const processedImages = await Promise.all(ACTOR_IMAGES_SIZE[type].map(processImageSize))
await remove(imagePhysicalFile.path)
return retryTransactionWrapper(() => sequelizeTypescript.transaction(async t => {
const actorImagesInfo = processedImages.map(({ imageName, imageSize }) => ({
name: imageName,
fileUrl: null,
height: imageSize.height,
width: imageSize.width,
onDisk: true
}))
const updatedActor = await updateActorImages(accountOrChannel.Actor, type, actorImagesInfo, t)
await updatedActor.save({ transaction: t })
await sendUpdateActor(accountOrChannel, t)
return type === ActorImageType.AVATAR
? updatedActor.Avatars
: updatedActor.Banners
}))
}
export async function deleteLocalActorImageFile (accountOrChannel: MAccountDefault | MChannelDefault, type: ActorImageType_Type) {
return retryTransactionWrapper(() => {
return sequelizeTypescript.transaction(async t => {
const updatedActor = await deleteActorImages(accountOrChannel.Actor, type, t)
await updatedActor.save({ transaction: t })
await sendUpdateActor(accountOrChannel, t)
return updatedActor.Avatars
})
})
}
// ---------------------------------------------------------------------------
2023-06-06 15:59:51 +02:00
export async function findAvailableLocalActorName (baseActorName: string, transaction?: Transaction) {
let actor = await ActorModel.loadLocalByName(baseActorName, transaction)
if (!actor) return baseActorName
for (let i = 1; i < 30; i++) {
const name = `${baseActorName}-${i}`
actor = await ActorModel.loadLocalByName(name, transaction)
if (!actor) return name
}
throw new Error('Cannot find available actor local name (too much iterations).')
}