deemix-js/deemix/tagger.js

315 lines
12 KiB
JavaScript
Raw Normal View History

const ID3Writer = require('./utils/id3-writer.js')
2021-04-18 11:02:47 +00:00
const Metaflac = require('metaflac-js2')
2021-04-14 15:11:56 +00:00
const fs = require('fs')
function tagID3(path, track, save){
const songBuffer = fs.readFileSync(path)
const tag = new ID3Writer(songBuffer)
tag.separateWithNull = String.fromCharCode(0)
2021-04-14 15:11:56 +00:00
if (save.title) tag.setFrame('TIT2', track.title)
if (save.artist && track.artists.length){
if (save.multiArtistSeparator == "default"){
tag.setFrame('TPE1', track.artists)
2021-04-14 15:11:56 +00:00
}else{
if (save.multiArtistSeparator == "nothing"){
tag.setFrame('TPE1', [track.mainArtist.name])
2021-04-14 15:11:56 +00:00
} else {
tag.setFrame('TPE1', [track.artistsString])
2021-04-14 15:11:56 +00:00
}
// Tag ARTISTS is added to keep the multiartist support when using a non standard tagging method
// https://picard-docs.musicbrainz.org/en/appendices/tag_mapping.html#artists
2021-12-19 16:27:48 +00:00
if (save.artists){
tag.setFrame('TXXX', {
description: 'ARTISTS',
value: track.artists
})
}
2021-04-14 15:11:56 +00:00
}
}
if (save.album) tag.setFrame('TALB', track.album.title)
if (save.albumArtist && track.album.artists.length){
if (save.singleAlbumArtist && track.album.mainArtist.save){
tag.setFrame('TPE2', [track.album.mainArtist.name])
2021-04-14 15:11:56 +00:00
} else {
tag.setFrame('TPE2', track.album.artists)
2021-04-14 15:11:56 +00:00
}
}
if (save.trackNumber){
let trackNumber = String(track.trackNumber)
2021-06-05 16:29:54 +00:00
if (save.trackTotal) trackNumber += `/${track.album.trackTotal}`
2021-04-14 15:11:56 +00:00
tag.setFrame('TRCK', trackNumber)
}
if (save.discNumber){
let discNumber = String(track.discNumber)
if (save.discTotal) discNumber += `/${track.album.discTotal}`
tag.setFrame('TPOS', discNumber)
}
if (save.genre) tag.setFrame('TCON', track.album.genre)
2021-04-14 15:11:56 +00:00
if (save.year) tag.setFrame('TYER', track.date.year)
// Referencing ID3 standard
// https://id3.org/id3v2.3.0#TDAT
// The 'Date' frame is a numeric string in the DDMM format.
if (save.date) tag.setFrame('TDAT', "" + track.date.day + track.date.month)
if (save.length) tag.setFrame('TLEN', parseInt(track.duration)*1000)
2021-09-30 17:40:54 +00:00
if (save.bpm && track.bpm) tag.setFrame('TBPM', track.bpm)
2021-04-14 15:11:56 +00:00
if (save.label) tag.setFrame('TPUB', track.album.label)
if (save.isrc) tag.setFrame('TSRC', track.ISRC)
if (save.barcode) tag.setFrame('TXXX', {
description: 'BARCODE',
value: track.album.barcode
})
if (save.explicit) tag.setFrame('TXXX', {
description: 'ITUNESADVISORY',
value: track.explicit ? "1" : "0"
})
if (save.replayGain) tag.setFrame('TXXX', {
description: 'REPLAYGAIN_TRACK_GAIN',
value: track.replayGain
})
2021-05-29 14:12:21 +00:00
if (save.lyrics && track.lyrics.unsync) tag.setFrame('USLT', {
description: '',
lyrics: track.lyrics.unsync,
language: 'XXX'
})
2021-04-14 15:11:56 +00:00
2021-06-18 08:07:27 +00:00
if (save.syncedLyrics && track.lyrics.syncID3.length !== 0) tag.setFrame('SYLT', {
2021-04-14 15:11:56 +00:00
text: track.lyrics.syncID3,
type: 1,
timestampFormat: 2,
useUnicodeEncoding: true
})
let involvedPeople = []
2021-04-18 11:02:47 +00:00
Object.keys(track.contributors).forEach(role => {
2021-04-21 17:00:51 +00:00
if (['author', 'engineer', 'mixer', 'producer', 'writer'].includes(role)){
2021-04-18 11:02:47 +00:00
track.contributors[role].forEach(person => {
2021-04-14 15:11:56 +00:00
involvedPeople.push([role, person])
})
} else if (role === 'composer' && save.composer){
tag.setFrame('TCOM', track.contributors.composer)
}
})
if (involvedPeople.length && save.involvedPeople) tag.setFrame('IPLS', involvedPeople)
2021-04-14 15:11:56 +00:00
if (save.copyright) tag.setFrame('TCOP', track.copyright)
if (save.savePlaylistAsCompilation && track.playlist || track.album.recordType == "compile")
tag.setFrame('TCMP', '1')
2021-04-14 15:11:56 +00:00
if (save.source){
tag.setFrame('TXXX', {
description: 'SOURCE',
value: 'Deezer'
})
tag.setFrame('TXXX', {
description: 'SOURCEID',
value: track.id
})
}
2021-09-21 11:54:02 +00:00
if (save.rating) {
let rank = (track.rank / 10000) * 2.55;
rank = rank > 255 ? 255 : Math.round(rank);
tag.setFrame('POPM', {
rating: rank
})
}
2021-04-14 15:11:56 +00:00
if (save.cover && track.album.embeddedCoverPath){
2021-04-14 16:10:31 +00:00
const coverArrayBuffer = fs.readFileSync(track.album.embeddedCoverPath)
2021-06-19 10:02:58 +00:00
if (coverArrayBuffer.length != 0){
tag.setFrame('APIC', {
type: 3,
data: coverArrayBuffer,
description: 'cover',
useUnicodeEncoding: save.coverDescriptionUTF8
})
}
2021-04-14 15:11:56 +00:00
}
tag.addTag()
2021-04-14 16:10:31 +00:00
2021-09-10 17:25:37 +00:00
let taggedSongBuffer = Buffer.from(tag.arrayBuffer)
if (taggedSongBuffer.slice(-128,-125).toString() === "TAG")
taggedSongBuffer = taggedSongBuffer.slice(0, -128)
if (save.saveID3v1)
taggedSongBuffer = tagID3v1(taggedSongBuffer, track, save)
fs.writeFileSync(path, taggedSongBuffer)
2021-04-14 15:11:56 +00:00
}
2021-04-18 11:02:47 +00:00
function tagFLAC(path, track, save){
const flac = new Metaflac(path)
flac.removeAllTags()
if (save.title) flac.setTag(`TITLE=${track.title}`)
if (save.artist && track.artists.length){
if (save.multiArtistSeparator == "default"){
track.artists.forEach(artist => {
flac.setTag(`ARTIST=${artist}`)
})
}else{
if (save.multiArtistSeparator == "nothing"){
flac.setTag(`ARTIST=${track.mainArtist.name}`)
} else {
flac.setTag(`ARTIST=${track.artistsString}`)
2021-04-18 11:02:47 +00:00
}
// Tag ARTISTS is added to keep the multiartist support when using a non standard tagging method
// https://picard-docs.musicbrainz.org/en/appendices/tag_mapping.html#artists
2021-12-19 16:27:48 +00:00
if (save.artists){
track.artists.forEach(artist => {
flac.setTag(`ARTISTS=${artist}`)
})
}
2021-04-18 11:02:47 +00:00
}
}
if (save.album) flac.setTag(`ALBUM=${track.album.title}`)
if (save.albumArtist && track.album.artists.length){
if (save.singleAlbumArtist && track.album.mainArtist.save){
flac.setTag(`ALBUMARTIST=${track.album.mainArtist.name}`)
} else {
track.album.artists.forEach(artist => {
flac.setTag(`ALBUMARTIST=${artist}`)
})
}
}
if (save.trackNumber) flac.setTag(`TRACKNUMBER=${track.trackNumber}`)
if (save.trackTotal) flac.setTag(`TRACKTOTAL=${track.album.trackTotal}`)
if (save.discNumber) flac.setTag(`DISCNUMBER=${track.discNumber}`)
if (save.discTotal) flac.setTag(`DISCTOTAL=${track.album.discTotal}`)
if (save.genre){
track.album.genre.forEach(genre => {
flac.setTag(`GENRE=${genre}`)
})
}
// YEAR tag is not suggested as a standard tag
// Being YEAR already contained in DATE will only use DATE instead
// Reference: https://www.xiph.org/vorbis/doc/v-comment.html#fieldnames
if (save.date) flac.setTag(`DATE=${track.dateString}`)
else if (save.year) flac.setTag(`DATE=${track.date.year}`)
if (save.length) flac.setTag(`LENGTH=${parseInt(track.duration)*1000}`)
2021-09-30 17:40:54 +00:00
if (save.bpm && track.bpm) flac.setTag(`BPM=${track.bpm}`)
2021-04-18 11:02:47 +00:00
if (save.label) flac.setTag(`PUBLISHER=${track.album.label}`)
if (save.isrc) flac.setTag(`ISRC=${track.ISRC}`)
if (save.barcode) flac.setTag(`BARCODE=${track.album.barcode}`)
if (save.explicit) flac.setTag(`ITUNESADVISORY=${track.explicit ? "1" : "0"}`)
if (save.replayGain) flac.setTag(`REPLAYGAIN_TRACK_GAIN=${track.replayGain}`)
if (save.lyrics && track.lyrics.unsync) flac.setTag(`LYRICS=${track.lyrics.unsync}`)
Object.keys(track.contributors).forEach(role => {
2021-04-21 17:00:51 +00:00
if (['author', 'engineer', 'mixer', 'producer', 'writer', 'composer'].includes(role)){
2021-04-18 11:02:47 +00:00
if (save.involvedPeople && role != 'composer' || save.composer && role == 'composer')
track.contributors[role].forEach(person => {
flac.setTag(`${role.toUpperCase()}=${person}`)
})
} else if (role === 'musicpublisher' && save.involvedPeople){
track.contributors.musicpublisher.forEach(person => {
flac.setTag(`ORGANIZATION=${person}`)
})
}
})
if (save.copyright) flac.setTag(`COPYRIGHT=${track.copyright}`)
if (save.savePlaylistAsCompilation && track.playlist || track.album.recordType == "compile")
flac.setTag('COMPILATION=1')
if (save.source){
flac.setTag('SOURCE=Deezer')
flac.setTag(`SOURCEID=${track.id}`)
}
2021-09-21 11:54:02 +00:00
if (save.rating) {
let rank = Math.round(track.rank / 10000);
flac.setTag(`RATING=${rank}`)
}
2021-04-18 11:02:47 +00:00
if (save.cover && track.album.embeddedCoverPath){
let picture = fs.readFileSync(track.album.embeddedCoverPath)
if (picture.length != 0) flac.importPicture(picture)
2021-04-18 11:02:47 +00:00
}
flac.save()
}
2021-05-28 10:00:40 +00:00
const id3v1Genres = ["Blues", "Classic Rock", "Country", "Dance", "Disco", "Funk", "Grunge", "Hip-Hop", "Jazz", "Metal", "New Age", "Oldies", "Other", "Pop", "Rhythm and Blues", "Rap", "Reggae", "Rock", "Techno", "Industrial", "Alternative", "Ska", "Death Metal", "Pranks", "Soundtrack", "Euro-Techno", "Ambient", "Trip-Hop", "Vocal", "Jazz & Funk", "Fusion", "Trance", "Classical", "Instrumental", "Acid", "House", "Game", "Sound clip", "Gospel", "Noise", "Alternative Rock", "Bass", "Soul", "Punk", "Space", "Meditative", "Instrumental Pop", "Instrumental Rock", "Ethnic", "Gothic", "Darkwave", "Techno-Industrial", "Electronic", "Pop-Folk", "Eurodance", "Dream", "Southern Rock", "Comedy", "Cult", "Gangsta", "Top 40", "Christian Rap", "Pop/Funk", "Jungle music", "Native US", "Cabaret", "New Wave", "Psychedelic", "Rave", "Showtunes", "Trailer", "Lo-Fi", "Tribal", "Acid Punk", "Acid Jazz", "Polka", "Retro", "Musical", "Rock n Roll", "Hard Rock", "Folk", "Folk-Rock", "National Folk", "Swing", "Fast Fusion", "Bebop", "Latin", "Revival", "Celtic", "Bluegrass", "Avantgarde", "Gothic Rock", "Progressive Rock", "Psychedelic Rock", "Symphonic Rock", "Slow Rock", "Big Band", "Chorus", "Easy Listening", "Acoustic", "Humour", "Speech", "Chanson", "Opera", "Chamber Music", "Sonata", "Symphony", "Booty Bass", "Primus", "Porn Groove", "Satire", "Slow Jam", "Club", "Tango", "Samba", "Folklore", "Ballad", "Power Ballad", "Rhythmic Soul", "Freestyle", "Duet", "Punk Rock", "Drum Solo", "A cappella", "Euro-House", "Dance Hall", "Goa music", "Drum & Bass", "Club-House", "Hardcore Techno", "Terror", "Indie", "BritPop", "Negerpunk", "Polsk Punk", "Beat", "Christian Gangsta Rap", "Heavy Metal", "Black Metal", "Crossover", "Contemporary Christian", "Christian Rock", "Merengue", "Salsa", "Thrash Metal", "Anime", "Jpop", "Synthpop", "Abstract", "Art Rock", "Baroque", "Bhangra", "Big beat", "Breakbeat", "Chillout", "Downtempo", "Dub", "EBM", "Eclectic", "Electro", "Electroclash", "Emo", "Experimental", "Garage", "Global", "IDM", "Illbient", "Industro-Goth", "Jam Band", "Krautrock", "Leftfield", "Lounge", "Math Rock", "New Romantic", "Nu-Breakz", "Post-Punk", "Post-Rock", "Psytrance", "Shoegaze", "Space Rock", "Trop Rock", "World Music", "Neoclassical", "Audiobook", "Audio Theatre", "Neue Deutsche Welle", "Podcast", "Indie-Rock", "G-Funk", "Dubstep", "Garage Rock", "Psybient"]
// Filters only Extended Ascii characters
function extAsciiFilter(string){
let output = ""
string.split('').forEach((x)=>{
if (x.charCodeAt(0) > 255)
output += "?"
else
output += x
})
return output
}
2021-09-10 17:25:37 +00:00
function tagID3v1(taggedSongBuffer, track, save){
2021-05-28 10:00:40 +00:00
const tagBuffer = Buffer.alloc(128)
tagBuffer.write('TAG', 0) // Header
if (save.title){
let trimmedTitle = extAsciiFilter(track.title.substring(0, 30))
tagBuffer.write(trimmedTitle, 3)
}
if (save.artist){
let selectedArtist
if (track.artistsString) selectedArtist = track.artistsString
2021-05-28 10:00:40 +00:00
else selectedArtist = track.mainArtist.name
let trimmedArtist = extAsciiFilter(selectedArtist.substring(0, 30))
tagBuffer.write(trimmedArtist, 33)
}
if (save.album){
let trimmedAlbum = extAsciiFilter(track.album.title.substring(0, 30))
tagBuffer.write(trimmedAlbum, 63)
}
if (save.year){
let trimmedYear = track.date.year.substring(0,4)
tagBuffer.write(trimmedYear, 93)
}
if (save.trackNumber){
if (track.trackNumber <= 65535){
if (track.trackNumber > 255){
tagBuffer.writeUInt8(parseInt(track.trackNumber >> 8), 125)
tagBuffer.writeUInt8(parseInt(track.trackNumber & 255), 126)
}else{
tagBuffer.writeUInt8(parseInt(track.trackNumber), 126)
}
}
}
if (save.genre){
let selectedGenre = track.album.genre[0]
if (id3v1Genres.includes(selectedGenre)) tagBuffer.writeUInt8(id3v1Genres.indexOf(selectedGenre), 127)
else tagBuffer.writeUInt8(255, 127)
}else{
tagBuffer.writeUInt8(255, 127)
}
// Save tags
2021-09-10 17:25:37 +00:00
const buffer = new ArrayBuffer(taggedSongBuffer.byteLength + 128)
2021-05-28 10:00:40 +00:00
const bufferWriter = new Uint8Array(buffer)
2021-09-10 17:25:37 +00:00
bufferWriter.set(new Uint8Array(taggedSongBuffer), 0)
bufferWriter.set(new Uint8Array(tagBuffer), taggedSongBuffer.byteLength)
return Buffer.from(buffer)
2021-05-28 10:00:40 +00:00
}
2021-04-14 15:11:56 +00:00
module.exports = {
2021-04-18 11:02:47 +00:00
tagID3,
2021-05-28 10:00:40 +00:00
tagFLAC,
tagID3v1
2021-04-14 15:11:56 +00:00
}