backend/utils/ytdlp.js

82 lines
2.9 KiB
JavaScript
Raw Normal View History

2024-03-29 17:48:59 +00:00
const wget = require('wget-improved')
2023-09-30 15:32:32 +00:00
const DOMPurify = require('isomorphic-dompurify')
const metadata = require('./metadata.js')
2024-03-29 17:48:59 +00:00
const hr = require('@tsmx/human-readable')
2023-03-03 16:44:40 +00:00
async function downloadVideo(url, ws, id) {
return new Promise(async (resolve, reject) => {
2024-03-30 17:04:00 +00:00
let quality = '720'
const video = await metadata.getVideoMetadata(id)
if (video.error) {
2024-03-30 13:32:37 +00:00
return resolve({
message: `Failed to request Youtube with error ${video.error}. Please retry...`,
fail: true
})
}
2024-03-30 17:04:00 +00:00
if (video.basic_info.duration >= 900) quality = '360' // 15 minutes
2024-03-29 17:48:59 +00:00
const downloadJson = await metadata.getVideoDownload(url, quality)
2024-03-29 18:30:04 +00:00
if (downloadJson.status == 'error') {
2024-03-30 13:32:37 +00:00
return resolve({
2024-03-29 18:30:04 +00:00
message: 'Failed to request Youtube. Please retry...',
fail: true
})
}
2024-03-29 17:48:59 +00:00
let size = ''
2024-03-29 18:30:04 +00:00
let startTime = Date.now()
let prevBytes = 0
let speed = 0
2024-03-29 17:48:59 +00:00
const alreadyPrecentages = []
2024-03-30 13:55:10 +00:00
const download = wget.download(downloadJson.url, `./videos/${id}.mp4`)
2023-03-03 16:44:40 +00:00
2024-03-29 17:48:59 +00:00
download.on('start', fileSize => {
size = fileSize
if (ws) ws.send(`DATA - Download has started in ${quality}`)
2023-03-03 16:44:40 +00:00
})
2024-03-29 17:48:59 +00:00
download.on('progress', progress => {
if (alreadyPrecentages.includes((progress*100).toFixed(0))) return
alreadyPrecentages.push((progress*100).toFixed(0))
2024-03-29 18:30:04 +00:00
const currentTime = Date.now()
const elapsedTime = (currentTime - startTime) / 1000
const currentBytes = progress * size
const bytesDownloaded = currentBytes - prevBytes
speed = bytesDownloaded / elapsedTime
prevBytes = currentBytes
const speedInMBps = speed / 1048576
const remainingBytes = size - currentBytes
const remainingTime = remainingBytes / speed
if (ws) ws.send(`DATA - [download] ${(progress*100).toFixed(2)}% of ${hr.fromBytes(size)} at ${speedInMBps.toFixed(2)} MB/s ETA ${secondsToTime(remainingTime.toFixed(0))}`)
})
2024-03-29 17:48:59 +00:00
download.on('error', err => {
if (ws) ws.send(`DATA - ${DOMPurify.sanitize(err)}`)
})
2024-03-29 17:48:59 +00:00
download.on('end', output => {
if (output == 'Finished writing to disk') {
ws.send(`DATA - Download has finished`)
2024-03-30 13:32:37 +00:00
return resolve({
2024-03-29 17:48:59 +00:00
fail: false
})
} else {
2024-03-30 13:32:37 +00:00
return resolve({
fail: true
})
}
})
})
}
2024-03-29 18:30:04 +00:00
function secondsToTime(seconds) {
const minutes = Math.floor(seconds / 60);
const remainingSeconds = seconds % 60;
const formattedSeconds = remainingSeconds < 10 ? '0' + remainingSeconds : remainingSeconds;
return `${minutes}:${formattedSeconds}`;
}
2023-03-03 16:44:40 +00:00
module.exports = { downloadVideo }