2023-12-29 19:55:56 +03:00
|
|
|
/*
|
2024-01-31 23:44:45 +03:00
|
|
|
* bytebeat-render.js [Node]
|
|
|
|
*
|
|
|
|
* Node.JS: Bytebeat generator
|
|
|
|
*
|
|
|
|
* Note: Calls `ffmpeg` to convert output raw audio to a WAVE file
|
|
|
|
*
|
|
|
|
* Warning: The current result is quick and dirty. Not for educational or
|
|
|
|
* production purposes.
|
|
|
|
*
|
2024-03-17 19:16:41 +03:00
|
|
|
* TODO: Fix signed bytebeat and floatbeat
|
2024-07-12 03:12:15 +03:00
|
|
|
* TODO: Add support for arbitrary (1 <= integer <= 2^32 - 1, because of
|
|
|
|
* WAVE/RIFF format limitations) audio bit depth
|
2024-01-31 23:44:45 +03:00
|
|
|
*
|
|
|
|
* Author: Intel A80486DX2-66
|
|
|
|
* License: Creative Commons Zero 1.0 Universal
|
|
|
|
*/
|
2023-12-29 19:55:56 +03:00
|
|
|
|
|
|
|
const { appendFileSync, unlinkSync, writeFileSync } = require("fs")
|
|
|
|
const { tmpdir } = require("os")
|
|
|
|
const { execSync } = require("child_process")
|
|
|
|
const { basename } = require("path")
|
|
|
|
|
|
|
|
let BUFFER_SIZE = 65536 // feel free to change this
|
2024-01-01 01:42:17 +03:00
|
|
|
const LIGHTNING_MODE = false // disables sequential file write optimization,
|
2024-01-31 22:30:50 +03:00
|
|
|
// feel free to enable this
|
2023-12-29 19:55:56 +03:00
|
|
|
|
|
|
|
const SAMPLE_RATE = 8000 // feel free to change this
|
|
|
|
const SECONDS = 30 // feel free to change this
|
|
|
|
const CHANNELS = 1 // feel free to change this
|
|
|
|
|
|
|
|
const FINAL_SAMPLE_RATE = 44100 // feel free to change this
|
|
|
|
const FINAL_SAMPLE_RATE_CONVERSION = SAMPLE_RATE / FINAL_SAMPLE_RATE / CHANNELS
|
|
|
|
const SAMPLES = SECONDS * FINAL_SAMPLE_RATE // feel free to change this
|
|
|
|
const PRODUCT = SAMPLES * CHANNELS
|
|
|
|
BUFFER_SIZE = LIGHTNING_MODE ? PRODUCT : BUFFER_SIZE
|
|
|
|
|
|
|
|
const TYPE_BYTEBEAT = 0
|
|
|
|
const TYPE_SIGNED_BYTEBEAT = 1
|
|
|
|
const TYPE_FLOATBEAT = 2
|
|
|
|
|
|
|
|
const SELECTED_TYPE = TYPE_BYTEBEAT // feel free to change this
|
|
|
|
|
|
|
|
// for bytebeat
|
2024-07-12 03:13:19 +03:00
|
|
|
var int = x => Math.floor(x)
|
|
|
|
var abs = Math.abs
|
|
|
|
var acos = Math.acos
|
|
|
|
var acosh = Math.acosh
|
|
|
|
var asin = Math.asin
|
|
|
|
var asinh = Math.asinh
|
|
|
|
var atan = Math.atan
|
|
|
|
var atanh = Math.atanh
|
|
|
|
var cbrt = Math.cbrt
|
|
|
|
var cos = Math.cos
|
|
|
|
var cosh = Math.cosh
|
|
|
|
var floor = Math.floor
|
|
|
|
var log = Math.log
|
|
|
|
var log2 = Math.log2
|
2024-07-12 03:14:00 +03:00
|
|
|
var max = Math.max
|
|
|
|
var min = Math.min
|
2024-07-12 03:13:19 +03:00
|
|
|
var PI = Math.PI
|
|
|
|
var pow = Math.pow
|
|
|
|
var random = Math.random
|
|
|
|
var sin = Math.sin
|
|
|
|
var sinh = Math.sinh
|
|
|
|
var sqrt = Math.sqrt
|
|
|
|
var tan = Math.tan
|
|
|
|
var tanh = Math.tanh
|
2023-12-29 19:55:56 +03:00
|
|
|
|
|
|
|
const generateAudio = t => {
|
|
|
|
return t&t>>8
|
|
|
|
}
|
|
|
|
|
2024-07-12 02:35:05 +03:00
|
|
|
let lastCorrectSample = 127 // FIXME: guessed value
|
|
|
|
|
2023-12-29 19:55:56 +03:00
|
|
|
const constrainValue = sample => {
|
2024-07-12 02:35:05 +03:00
|
|
|
if (isNaN(sample) || sample < 0)
|
|
|
|
sample = lastCorrectSample
|
|
|
|
else
|
|
|
|
lastCorrectSample = sample
|
|
|
|
|
2023-12-29 19:55:56 +03:00
|
|
|
switch (SELECTED_TYPE) {
|
|
|
|
case TYPE_BYTEBEAT:
|
|
|
|
return sample & 255
|
|
|
|
case TYPE_SIGNED_BYTEBEAT:
|
|
|
|
return ((sample + 127) & 255) - 128
|
|
|
|
case TYPE_FLOATBEAT:
|
|
|
|
return floor(sample * 127) + 127
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const random_choice = choices => choices[Math.floor(Math.random() * choices.length)]
|
|
|
|
|
2024-01-01 01:42:17 +03:00
|
|
|
const randomFileNameAlphabet =
|
|
|
|
"ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789_-"
|
2023-12-29 19:55:56 +03:00
|
|
|
|
2024-01-05 20:12:57 +03:00
|
|
|
const generateRandomFilePath = () => {
|
2023-12-29 19:55:56 +03:00
|
|
|
let res = tmpdir() + "/" + basename(__filename) + "_"
|
|
|
|
for (let i = 0; i < 64; i++)
|
|
|
|
res += random_choice(randomFileNameAlphabet)
|
|
|
|
return res
|
|
|
|
}
|
|
|
|
|
|
|
|
let t = 0
|
|
|
|
|
2024-01-05 20:12:57 +03:00
|
|
|
let filePath = generateRandomFilePath()
|
|
|
|
writeFileSync(filePath, Buffer.alloc(0))
|
2023-12-29 19:55:56 +03:00
|
|
|
|
|
|
|
// the loop of sequential file writing, created to ease load on RAM
|
2024-07-12 03:14:00 +03:00
|
|
|
const buffer_max = Math.floor((PRODUCT + (BUFFER_SIZE - 1)) / BUFFER_SIZE),
|
|
|
|
needTwoBuffers = buffer_max > 1, needSingleBuffer = !needTwoBuffers
|
2023-12-29 19:55:56 +03:00
|
|
|
|
2024-01-31 23:41:49 +03:00
|
|
|
let audioData = new Uint8Array(needSingleBuffer ? PRODUCT : BUFFER_SIZE)
|
2024-01-31 22:31:55 +03:00
|
|
|
|
2024-07-12 03:14:00 +03:00
|
|
|
for (let seq = 0; seq < buffer_max; seq++) {
|
2024-01-31 23:41:49 +03:00
|
|
|
if (needTwoBuffers && (t + BUFFER_SIZE) >= PRODUCT) {
|
|
|
|
let calculatedSize = PRODUCT - t
|
|
|
|
audioData = new Uint8Array(calculatedSize)
|
|
|
|
}
|
2024-01-31 22:31:55 +03:00
|
|
|
|
|
|
|
for (let idx = 0; t < PRODUCT && idx < BUFFER_SIZE; idx++, t++) {
|
2023-12-29 22:31:26 +03:00
|
|
|
let sample = generateAudio(t * FINAL_SAMPLE_RATE_CONVERSION)
|
2023-12-29 19:55:56 +03:00
|
|
|
if (sample.constructor === Array)
|
|
|
|
sample.forEach((sample, index) => {
|
2023-12-30 16:35:28 +03:00
|
|
|
audioData[CHANNELS * idx + index] = constrainValue(sample)
|
2023-12-29 19:55:56 +03:00
|
|
|
})
|
|
|
|
else
|
2023-12-30 16:35:28 +03:00
|
|
|
audioData[idx] = constrainValue(sample)
|
2023-12-29 19:55:56 +03:00
|
|
|
}
|
|
|
|
|
2024-01-05 20:12:57 +03:00
|
|
|
appendFileSync(filePath, Buffer.from(audioData.buffer))
|
2023-12-29 19:55:56 +03:00
|
|
|
}
|
|
|
|
|
2024-01-01 01:42:17 +03:00
|
|
|
execSync(
|
|
|
|
`ffmpeg -f u8 -ar ${FINAL_SAMPLE_RATE} -ac ${CHANNELS} ` +
|
2024-01-05 20:12:57 +03:00
|
|
|
`-i ${filePath} output_${+new Date()}.wav`)
|
|
|
|
unlinkSync(filePath)
|