deemix-js/deemix/utils/index.js

154 lines
5.2 KiB
JavaScript
Raw Normal View History

2021-04-14 16:10:31 +00:00
const stream = require('stream')
const {promisify} = require('util')
const pipeline = promisify(stream.pipeline)
2021-04-27 17:25:14 +00:00
const { accessSync, constants } = require('fs')
2021-08-02 21:45:19 +00:00
const { ErrorMessages } = require('../errors.js')
2021-04-14 16:10:31 +00:00
const USER_AGENT_HEADER = "Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/79.0.3945.130 Safari/537.36"
2021-04-27 17:25:14 +00:00
function canWrite(path){
try{
accessSync(path, constants.R_OK | constants.W_OK)
}catch{
return false
}
return true
}
2021-04-01 11:38:59 +00:00
function generateReplayGainString(trackGain){
return `${Math.round((parseFloat(trackGain) + 18.4)*-100)/100} dB`
2021-04-01 11:38:59 +00:00
}
2021-04-29 16:52:52 +00:00
function changeCase(txt, type){
switch (type) {
case 'lower': return txt.toLowerCase()
case 'upper': return txt.toUpperCase()
case 'start':
txt = txt.split(" ")
for (let i = 0, q = 0; i < txt.length; i++)
if (txt[i].substr(0, 1) === "(" || txt[i].substr(0, 1) === "[" || txt[i].substr(0, 1) === "{") q++;
txt[i] = txt[i][q].toUpperCase() + txt[i].substr(q + 1).toLowerCase()
2021-04-29 16:52:52 +00:00
return txt.join(" ")
case 'sentence': return txt.charAt(0).toUpperCase() + txt.slice(1).toLowerCase()
default: return txt
}
}
2021-04-01 11:38:59 +00:00
function removeFeatures(title){
let clean = title
if (clean.search(/\(feat\./gi) != -1){
const pos = clean.search(/\(feat\./gi)
2021-04-21 17:00:51 +00:00
let tempTrack = clean.slice(0, pos)
if (clean.includes(')'))
tempTrack += clean.slice(clean.indexOf(')', pos+1)+1)
2021-04-01 11:38:59 +00:00
clean = tempTrack.trim()
clean = clean.replace(/\s\s+/g, ' ') // remove extra spaces
}
return clean
}
function andCommaConcat(lst){
const tot = lst.length
let result = ""
lst.forEach((art, i) => {
result += art
if (tot != i+1){
if (tot - 1 == i+1){
result += " & "
} else {
result += ", "
}
}
})
return result
}
function uniqueArray(arr){
arr.forEach((namePrinc, iPrinc) => {
arr.forEach((nameRest, iRest) => {
2021-04-21 17:00:51 +00:00
if (iPrinc != iRest && nameRest.toLowerCase().includes(namePrinc.toLowerCase())){
2021-04-01 11:38:59 +00:00
arr.splice(iRest, 1)
}
})
})
return arr
}
function shellEscape(s){
if (typeof s !== 'string') return ''
if (!(/[^\w@%+=:,./-]/g.test(s))) return s
return "'" + s.replaceAll("'", "'\"'\"'") + "'"
}
2021-04-01 11:38:59 +00:00
function removeDuplicateArtists(artist, artists){
artists = uniqueArray(artists)
Object.keys(artist).forEach((role) => {
artist[role] = uniqueArray(artist[role])
})
return [artist, artists]
}
2021-07-27 19:21:08 +00:00
function formatListener(key, data){
let message = ""
switch (key) {
case "startAddingArtist": return `Started gathering ${data.name}'s albums (${data.id})`
case "finishAddingArtist": return `Finished gathering ${data.name}'s albums (${data.id})`
2021-08-02 21:45:19 +00:00
case "updateQueue":
2021-08-02 21:54:19 +00:00
message = `[${data['uuid']}]`
if (data.downloaded) message += ` Completed download of ${data.downloadPath.slice(data.extrasPath.length)}`
2021-08-02 21:45:19 +00:00
if (data.failed) message += ` ${data.data.artist} - ${data.data.title} :: ${data.error}`
if (data.progress) message += ` Download at ${data.progress}%`
if (data.conversion) message += ` Conversion at ${data.conversion}%`
return message
2021-07-27 19:21:08 +00:00
case "downloadInfo":
2021-08-02 21:45:19 +00:00
message = data.state
switch (data.state) {
case "getTags": message = "Getting tags."; break;
case "gotTags": message = "Tags got."; break;
case "getBitrate": message = "Getting download URL."; break;
case "bitrateFallback": message = "Desired bitrate not found, falling back to lower bitrate."; break;
case "searchFallback": message = "This track has been searched for, result might not be 100% exact."; break;
case "gotBitrate": message = "Download URL got."; break;
case "getAlbumArt": message = "Downloading album art."; break;
case "gotAlbumArt": message = "Album art downloaded."; break;
case "downloading":
message = "Downloading track.";
if (data.alreadyStarted) message += ` Recovering download from ${data.value}.`
else message += ` Downloading ${data.value} bytes.`
break;
case "downloaded": message = "Track downloaded."; break;
case "alreadyDownloaded": message = "Track already downloaded."; break;
case "tagging": message = "Tagging track."; break;
case "tagged": message = "Track tagged."; break;
}
return `[${data.uuid}] ${data.data.artist} - ${data.data.title} :: ${message}`
case "downloadWarn":
message = `[${data.uuid}] ${data.data.artist} - ${data.data.title} :: ${ErrorMessages[data.state]} `
switch (data.solution) {
case 'fallback': message += "Using fallback id."; break;
case 'search': message += "Searching for alternative."; break;
}
return message
2021-08-02 21:54:19 +00:00
case "currentItemCancelled": return `Current item cancelled (${data})`
case "removedFromQueue": return `[${data}] Removed from the queue`
case "finishDownload": return `[${data}] Finished downloading`
case "startConversion": return `[${data}] Started converting`
case "finishConversion": return `[${data}] Finished converting`
2021-08-02 21:45:19 +00:00
default: return message
2021-07-27 19:21:08 +00:00
}
}
2021-04-01 11:38:59 +00:00
module.exports = {
USER_AGENT_HEADER,
2021-04-01 11:38:59 +00:00
generateReplayGainString,
removeFeatures,
andCommaConcat,
uniqueArray,
2021-04-14 16:10:31 +00:00
removeDuplicateArtists,
2021-04-27 17:25:14 +00:00
pipeline,
2021-04-29 16:52:52 +00:00
canWrite,
changeCase,
2021-07-27 19:21:08 +00:00
shellEscape,
formatListener
2021-04-01 11:38:59 +00:00
}