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
|
|
|
|
2021-04-09 18:05:32 +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){
|
2021-04-09 16:45:57 +00:00
|
|
|
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':
|
2022-01-02 13:14:31 +00:00
|
|
|
txt = txt.trim().split(" ")
|
2021-09-01 03:45:16 +00:00
|
|
|
for (let i = 0; i < txt.length; i++) {
|
2022-01-02 13:14:31 +00:00
|
|
|
if (['(', '{', '[', "'", '"'].some(bracket => ( txt[i].length > 1 && txt[i].startsWith(bracket) ) )) {
|
2021-09-01 03:45:16 +00:00
|
|
|
txt[i] = txt[i][0] + txt[i][1].toUpperCase() + txt[i].substr(2).toLowerCase()
|
2022-01-02 13:14:31 +00:00
|
|
|
} else if (txt[i].length > 1) {
|
2021-09-01 03:45:16 +00:00
|
|
|
txt[i] = txt[i][0].toUpperCase() + txt[i].substr(1).toLowerCase()
|
2022-01-02 13:14:31 +00:00
|
|
|
} else {
|
|
|
|
txt[i] = txt[i][0].toUpperCase()
|
2021-09-01 03:45:16 +00:00
|
|
|
}
|
|
|
|
}
|
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
|
2021-11-12 19:58:52 +00:00
|
|
|
let found = false
|
|
|
|
let pos
|
2022-01-02 12:35:34 +00:00
|
|
|
if (clean.search(/[\s(]\(?\s?feat\.?\s/gi) != -1){
|
|
|
|
pos = clean.search(/[\s(]\(?\s?feat\.?\s/gi)
|
2021-11-12 19:58:52 +00:00
|
|
|
found = true
|
|
|
|
}
|
2022-01-02 12:35:34 +00:00
|
|
|
if (clean.search(/[\s(]\(?\s?ft\.?\s/gi) != -1){
|
|
|
|
pos = clean.search(/[\s(]\(?\s?ft\.?\s/gi)
|
2021-11-12 19:58:52 +00:00
|
|
|
found = true
|
|
|
|
}
|
2022-01-02 12:35:34 +00:00
|
|
|
const openBracket = clean[pos] == '(' || clean[pos+1] == '('
|
|
|
|
const otherBracket = clean.indexOf('(', pos+2)
|
2021-11-12 19:58:52 +00:00
|
|
|
if (found) {
|
2021-04-21 17:00:51 +00:00
|
|
|
let tempTrack = clean.slice(0, pos)
|
2021-11-12 19:58:52 +00:00
|
|
|
if (clean.includes(')') && openBracket)
|
2022-01-02 12:35:34 +00:00
|
|
|
tempTrack += clean.slice(clean.indexOf(')', pos+2)+1)
|
|
|
|
if (!openBracket && otherBracket != -1)
|
|
|
|
tempTrack += ` ${clean.slice(otherBracket)}`
|
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
|
|
|
|
}
|
|
|
|
|
2021-05-31 21:25:53 +00:00
|
|
|
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']}]`
|
2021-12-21 11:45:02 +00:00
|
|
|
if (data.downloaded) message += ` Completed download of ${data.downloadPath.slice(data.extrasPath.length+1)}`
|
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`
|
2021-09-21 07:20:50 +00:00
|
|
|
case "finishConversion": return `[${data.uuid}] 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 = {
|
2021-04-09 18:05:32 +00:00
|
|
|
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,
|
2021-05-31 21:25:53 +00:00
|
|
|
changeCase,
|
2021-07-27 19:21:08 +00:00
|
|
|
shellEscape,
|
|
|
|
formatListener
|
2021-04-01 11:38:59 +00:00
|
|
|
}
|