peertube-plugin-livechat/build-avatars.js

390 lines
11 KiB
JavaScript
Raw Normal View History

2023-04-07 11:08:20 +02:00
#!/bin/env node
2024-05-23 11:42:14 +02:00
// SPDX-FileCopyrightText: 2024 John Livingston <https://www.john-livingston.fr/>
//
// SPDX-License-Identifier: AGPL-3.0-only
2022-01-11 01:34:27 +01:00
/* eslint-env es6 */
2022-01-07 00:53:54 +01:00
const sharp = require('sharp')
const fs = require('node:fs')
const path = require('node:path')
const {
Worker, isMainThread, parentPort, workerData,
} = require('node:worker_threads')
const avatarPartsDef = {
// Available parts:
// Note: some part files are empty, so that David's generator don't always add every part.
// But this make my algorithm generate a lot of avatars that have no part other that the body and the yes.
// So i don't include all empty files
'sepia': {
body: 25,
pattern: 14, // 12 to 20 are empty
mouth: 10,
eyes: 10,
accessories: 17, // 14 to 20 are empty
misc: 16, // 15 to 20 are empty
hat: 20 // 13 to 20 are empty
},
'cat': {
body: 15,
fur: 10,
eyes: 15,
mouth: 10,
accessorie: 20 // 17 to 20 are empty
},
'bird': {
tail: 9, // here we must begin with the tail
hoop: 10,
body: 9,
wing: 9,
eyes: 9,
bec: 9,
accessorie: 16 // 15 to 20 are empty
},
'fenec': {
body: 25,
nose: 10,
tail: 5,
eyes: 10,
mouth: 10,
accessories: 16, // 14 to 20 are empty
misc: 17, // 15 to 20 are empty
hat: 15 // 13 to 20 are empty
},
2024-02-12 16:50:08 +01:00
'abstract': {
body: 15,
fur: 10,
eyes: 15,
mouth: 10
2024-08-05 21:49:12 -04:00
},
'nctv': {
body: null,
2024-02-12 16:50:08 +01:00
}
}
function generateLegacyAvatars () {
// Legacy avatars generation
const inputDir = './assets/images/avatars/legacy'
const outputDir = './dist/server/avatars/legacy'
fs.mkdirSync(outputDir, { recursive: true })
const backgrounds = [
'#ffffff',
'#000000',
'#ff0000',
'#00ff00',
'#0000ff',
'#808000',
'#ffff00',
'#008000',
'#008080',
'#00ffff',
'#000080',
'#800080',
'#ff00ff'
]
const count = 10
for (let i = 1; i <= count; i++) {
const inputFile = path.join(inputDir, i + '.svg')
for (let j = 0; j < backgrounds.length; j++) {
const out = i + (count * j)
const background = backgrounds[j]
sharp(inputFile).flatten({background}).resize(120, 120).jpeg({quality: 95, mozjpeg: true}).toFile(path.join(outputDir, out.toString() + '.jpg'))
}
}
// Moderation bot avatar: for now taking image 2, and applying a grey background.
{
const i = 2
const inputFile = path.join(inputDir, i + '.svg')
const background = '#858da0'
const outputDir = './dist/server/bot_avatars/legacy'
fs.mkdirSync(outputDir, { recursive: true })
const out = 1
2022-01-07 11:29:20 +01:00
sharp(inputFile).flatten({background}).resize(120, 120).jpeg({quality: 95, mozjpeg: true}).toFile(path.join(outputDir, out.toString() + '.jpg'))
2022-01-07 00:53:54 +01:00
}
}
// 2024 avatars generation
async function generateAvatars (part) {
console.log('Starting generating ' + part)
const parts = avatarPartsDef[part]
if (!parts) {
throw new Error('Missing part\'s conf: ' + part)
}
const inputDir = path.join('./assets/images/avatars/', part)
const outputDir = path.join('./dist/server/avatars/', part)
fs.mkdirSync(outputDir, { recursive: true })
function computeFilename (part, count) {
let a = (1 + (count % parts[part])).toString()
if (a.length < 2) { a = '0' + a}
return path.join(
inputDir,
part + '_' + a + '.png'
)
}
2024-08-05 21:46:02 -04:00
// We can't generate all combinations! It would make 400 000 000 files!
// So we arbitrary pick some combinations, using some modulus
const nb = 200 // number of avatars to generate
2024-08-05 21:46:02 -04:00
for (let i = 0; i < nb; i++) {
const ouputFile = path.join(
outputDir,
i.toString() + '.png'
)
if (await fs.existsSync(ouputFile)) {
console.log(`Skipping ${ouputFile}, file already exists`)
continue
}
2024-08-05 21:46:02 -04:00
const partsToCombine = Object.keys(parts)
const firstPart = partsToCombine.shift()
const firstFile = computeFilename(firstPart, i)
// We just have to combinate different parts into one file, then output at the wanted size.
const composites = []
let j = 0
for (const part of partsToCombine) {
j++ // introduce an offset so we don't get all empty parts at the same time
composites.push({
input: computeFilename(part, i + (j * 7))
})
}
2024-08-05 21:46:02 -04:00
const buff = await sharp(firstFile)
.composite(composites)
.toBuffer()
2024-08-05 21:46:02 -04:00
await sharp(buff)
.resize(60, 60)
.png({
compressionLevel: 9,
palette: true
})
.toFile(ouputFile)
}
}
2024-08-05 21:46:02 -04:00
const generateNigbotAvatar = async () => {
console.log('Starting generating nigbot avatar');
2024-08-05 21:46:02 -04:00
const inputDir = './assets/images/avatars/nctv/';
const outputDir = './dist/server/avatars/nctv/';
fs.mkdirSync(outputDir, { recursive: true });
2024-08-05 21:46:02 -04:00
const buff = await sharp(path.join(inputDir, 'nigbot.png')).toBuffer();
await sharp(buff)
.resize(60, 60)
.png({ palette: true })
.toFile(path.join(outputDir, '1.png'));
}
2024-02-09 13:17:12 +01:00
async function generateBotsAvatars () {
{
2024-07-10 11:55:54 +02:00
// Moderation bot avatar: choosing some parts, and turning it so it is facing left.
const inputDir = path.join('./assets/images/avatars/', 'sepia')
const botOutputDir = './dist/server/bot_avatars/sepia/'
fs.mkdirSync(botOutputDir, { recursive: true })
const buff = await sharp(path.join(inputDir, 'body_20.png'))
.composite([
{ input: path.join(inputDir, 'pattern_01.png') },
{ input: path.join(inputDir, 'mouth_01.png') },
{ input: path.join(inputDir, 'eyes_01.png') },
{ input: path.join(inputDir, 'misc_05.png') },
{ input: path.join(inputDir, 'hat_07.png') }
])
.toBuffer()
2024-02-09 13:17:12 +01:00
await sharp(buff)
.flop() // horizontal mirror
.resize(60, 60)
.png({
compressionLevel: 9,
palette: true
})
.toFile(path.join(botOutputDir, '1.png'))
}
{
2024-07-10 11:55:54 +02:00
// Moderation bot avatar: choosing some parts, and turning it so it is facing left.
const inputDir = path.join('./assets/images/avatars/', 'cat')
const botOutputDir = './dist/server/bot_avatars/cat/'
fs.mkdirSync(botOutputDir, { recursive: true })
const buff = await sharp(path.join(inputDir, 'body_04.png'))
.composite([
{ input: path.join(inputDir, 'mouth_02.png') },
{ input: path.join(inputDir, 'eyes_11.png') },
{ input: path.join(inputDir, 'fur_02.png') },
{ input: path.join(inputDir, 'accessorie_03.png') }
])
.toBuffer()
await sharp(buff)
.flop() // horizontal mirror
.resize(60, 60)
.png({
compressionLevel: 9,
palette: true
})
.toFile(path.join(botOutputDir, '1.png'))
}
{
2024-07-10 11:55:54 +02:00
// Moderation bot avatar: choosing some parts, and turning it so it is facing left.
const inputDir = path.join('./assets/images/avatars/', 'bird')
const botOutputDir = './dist/server/bot_avatars/bird/'
fs.mkdirSync(botOutputDir, { recursive: true })
const buff = await sharp(path.join(inputDir, 'tail_06.png'))
.composite([
{ input: path.join(inputDir, 'hoop_04.png')},
{ input: path.join(inputDir, 'body_07.png')},
{ input: path.join(inputDir, 'wing_03.png')},
{ input: path.join(inputDir, 'eyes_05.png')},
{ input: path.join(inputDir, 'bec_07.png')},
{ input: path.join(inputDir, 'accessorie_03.png')}
])
.toBuffer()
await sharp(buff)
.flop() // horizontal mirror
.resize(60, 60)
.png({
compressionLevel: 9,
palette: true
})
.toFile(path.join(botOutputDir, '1.png'))
}
2024-02-09 15:12:48 +01:00
{
2024-07-10 11:55:54 +02:00
// Moderation bot avatar: choosing some parts, and turning it so it is facing left.
const inputDir = './assets/images/avatars/fenec'
const botOutputDir = './dist/server/bot_avatars/fenec/'
fs.mkdirSync(botOutputDir, { recursive: true })
const buff = await sharp(path.join(inputDir, 'body_15.png'))
.composite([
{ input: path.join(inputDir, 'nose_07.png') },
{ input: path.join(inputDir, 'tail_04.png') },
{ input: path.join(inputDir, 'eyes_03.png') },
{ input: path.join(inputDir, 'mouth_07.png') },
{ input: path.join(inputDir, 'accessories_08.png') },
{ input: path.join(inputDir, 'misc_05.png') },
{ input: path.join(inputDir, 'hat_07.png') }
])
.toBuffer()
2024-02-09 15:12:48 +01:00
await sharp(buff)
.flop() // horizontal mirror
.resize(60, 60)
.png({
compressionLevel: 9,
palette: true
})
.toFile(path.join(botOutputDir, '1.png'))
}
2024-02-12 16:50:08 +01:00
{
2024-07-10 11:55:54 +02:00
// Moderation bot avatar: choosing some parts, and turning it so it is facing left.
2024-02-12 16:50:08 +01:00
const inputDir = './assets/images/avatars/abstract'
const botOutputDir = './dist/server/bot_avatars/abstract/'
fs.mkdirSync(botOutputDir, { recursive: true })
const buff = await sharp(path.join(inputDir, 'body_08.png'))
.composite([
{ input: path.join(inputDir, 'body_15.png') }, // here we add a second body
{ input: path.join(inputDir, 'fur_08.png') },
{ input: path.join(inputDir, 'mouth_03.png') }
])
.toBuffer()
await sharp(buff)
.flop() // horizontal mirror
.resize(60, 60)
.png({
compressionLevel: 9,
palette: true
})
.toFile(path.join(botOutputDir, '1.png'))
}
2024-08-05 20:29:46 -04:00
{
// Nigbot avatar for users
const inputDir = './assets/images/avatars/nctv'
const botOutputDir = './dist/server/bot_avatars/nctv/'
fs.mkdirSync(botOutputDir, { recursive: true })
const buff = await sharp(path.join(inputDir, 'nigbot.png'))
.toBuffer()
await sharp(buff)
2024-08-05 20:44:04 -04:00
// .resize(60, 60)
.png()
2024-08-05 20:29:46 -04:00
.toFile(path.join(botOutputDir, '1.png'))
}
}
2024-02-09 15:35:40 +01:00
if (isMainThread) {
const what = [
'legacy',
...Object.keys(avatarPartsDef),
'bots'
]
// Creating Worker threads to process each avatar type in parallel
for (const part of what) {
const p = new Promise((resolve, reject) => {
const worker = new Worker(__filename, {
workerData: part
})
worker.on('message', resolve)
worker.on('error', reject)
worker.on('exit', (code) => {
if (code !== 0) {
reject (new Error(`Worker stopped with exit code ${code}`))
}
})
})
2024-02-09 15:35:40 +01:00
p.then(
2024-02-12 15:19:58 +01:00
(msg) => console.log(part + ' avatars: ' + msg),
() => console.error
)
}
} else {
const part = workerData
if (part === 'legacy') {
generateLegacyAvatars()
parentPort.postMessage('done')
} else if (part === 'bots') {
generateBotsAvatars().then(
() => {
parentPort.postMessage('done')
},
(err) => {
throw err
}
)
2024-08-05 21:46:02 -04:00
} else if (part === 'nctv') {
generateNigbotAvatar();
parentPort.postMessage('done');
} else {
generateAvatars(part).then(
() => {
parentPort.postMessage('done')
},
(err) => {
throw err
}
)
}
}