deemix-js/deemix/plugins/spotify.js
2021-06-05 18:20:22 +02:00

388 lines
12 KiB
JavaScript

const Plugin = require('./plugin.js')
const { getConfigFolder } = require('../utils/localpaths.js')
const {
generateTrackItem,
generateAlbumItem,
TrackNotOnDeezer,
AlbumNotOnDeezer,
InvalidID
} = require('../itemgen.js')
const { Convertable, Collection } = require('../types/DownloadObjects.js')
const { sep } = require('path')
const fs = require('fs')
const SpotifyWebApi = require('spotify-web-api-node')
const got = require('got')
const { queue } = require('async')
class Spotify extends Plugin {
constructor(configFolder = undefined){
super()
this.credentials = {clientId: "", clientSecret: ""}
this.enabled = false
this.sp
this.configFolder = configFolder || getConfigFolder()
this.configFolder += `spotify${sep}`
return this
}
setup(){
fs.mkdirSync(this.configFolder, { recursive: true })
if (! fs.existsSync(this.configFolder+'credentials.json')) fs.writeFileSync(this.configFolder+'credentials.json', JSON.stringify(this.credentials))
this.credentials = JSON.parse(fs.readFileSync(this.configFolder+'credentials.json'))
this.checkCredentials()
return this
}
async parseLink(link){
if (link.includes('link.tospotify.com')){
link = await got.get(link) // Resolve URL shortner
link = link.url
}
// Remove extra stuff
if (link.includes('?')) link = link.slice(0, link.indexOf('?'))
if (link.includes('&')) link = link.slice(0, link.indexOf('&'))
if (link.endsWith('/')) link = link.slice(0, -1) // Remove last slash if present
let link_type, link_id
if (!link.includes('spotify')) return [link, link_type, link_id] // return if not a spotify link
if (link.search(/[/:]track[/:](.+)/g) != -1){
link_type = 'track'
link_id = /[/:]track[/:](.+)/g.exec(link)[1]
}else if (link.search(/[/:]album[/:](.+)/g) != -1){
link_type = 'album'
link_id = /[/:]album[/:](.+)/g.exec(link)[1]
}else if (link.search(/[/:]playlist[/:](\d+)/g) != -1){
link_type = 'playlist'
link_id = /[/:]playlist[/:](.+)/g.exec(link)[1]
}
return [link, link_type, link_id]
}
async generateDownloadObject(dz, link, bitrate){
let link_type, link_id
[link, link_type, link_id] = await this.parseLink(link)
if (link_type == null || link_id == null) return null
switch (link_type) {
case 'track':
return this.generateTrackItem(dz, link_id, bitrate)
case 'album':
return this.generateAlbumItem(dz, link_id, bitrate)
case 'playlist':
return this.generatePlaylistItem(dz, link_id, bitrate)
}
}
async generateTrackItem(dz, link_id, bitrate){
let [track_id, trackAPI] = await this.convertTrack(dz, link_id)
if (track_id !== "0"){
return generateTrackItem(dz, track_id, bitrate, trackAPI)
} else {
throw new TrackNotOnDeezer(`https://open.spotify.com/track/${link_id}`)
}
}
async generateAlbumItem(dz, link_id, bitrate){
let album_id = await this.convertAlbum(dz, link_id)
if (album_id !== "0"){
return generateAlbumItem(dz, album_id, bitrate)
} else {
throw new AlbumNotOnDeezer(`https://open.spotify.com/album/${link_id}`)
}
}
async generatePlaylistItem(dz, link_id, bitrate){
if (!this.enabled) throw new Error("Spotify plugin not enabled")
let spotifyPlaylist = await this.sp.getPlaylist(link_id)
spotifyPlaylist = spotifyPlaylist.body
let playlistAPI = this._convertPlaylistStructure(spotifyPlaylist)
playlistAPI.various_artist = await dz.api.get_artist(5080) // Useful for save as compilation
let tracklistTemp = spotifyPlaylist.tracks.items
while (spotifyPlaylist.tracks.next) {
let regExec = /offset=(\d+)&limit=(\d+)/g.exec(spotifyPlaylist.tracks.next)
let offset = regExec[1]
let limit = regExec[2]
let playlistTracks = await this.sp.getPlaylistTracks(link_id, { offset, limit })
spotifyPlaylist.tracks = playlistTracks.body
tracklistTemp = tracklistTemp.concat(spotifyPlaylist.tracks.items)
}
let tracklist = []
tracklistTemp.forEach((item) => {
if (!item.track) return // Skip everything that isn't a track
if (item.track.explicit && !playlistAPI.explicit) playlistAPI.explicit = true
tracklist.push(item.track)
})
if (!playlistAPI.explicit) playlistAPI.explicit = false
return new Convertable({
type: 'spotify_playlist',
id: link_id,
bitrate,
title: spotifyPlaylist.name,
artist: spotifyPlaylist.owner.display_name,
cover: playlistAPI.picture_thumbnail,
explicit: playlistAPI.explicit,
size: tracklist.length,
collection: {
tracks_gw: [],
playlistAPI: playlistAPI
},
plugin: 'spotify',
conversion_data: tracklist
})
}
async convertTrack(dz, track_id, fallbackSearch = false, cachedTrack = null){
if (!this.enabled) throw new Error("Spotify plugin not enabled")
let shouldSaveCache = false
let cache
if (!cachedTrack){
try {
cache = JSON.parse(fs.readFileSync(this.configFolder+'cache.json'))
} catch {
cache = {tracks: {}, albums: {}}
}
shouldSaveCache = true
if (cache.tracks[track_id]){
cachedTrack = cache.tracks[track_id]
} else {
try{
cachedTrack = await this.sp.getTrack(track_id)
} catch (e){
if (e.body.error.message === "invalid id") throw new InvalidID(`https://open.spotify.com/track/${track_id}`)
throw e
}
cachedTrack = cachedTrack.body
}
}
let dz_id = "0"
let dz_track = null
let isrc = null
if (cachedTrack.external_ids && cachedTrack.external_ids.isrc){
isrc = cachedTrack.external_ids.isrc
dz_track = await dz.api.get_track_by_ISRC(isrc)
if (dz_track.title && dz_track.id) dz_id = dz_track.id
}
if (dz_id === "0" && fallbackSearch){
dz_id = dz.api.get_track_id_from_metadata(
cachedTrack.artists[0].name,
cachedTrack.name,
cachedTrack.album.name
)
}
if (shouldSaveCache){
cache.tracks[track_id] = {id: dz_id, isrc: isrc}
fs.writeFileSync(this.configFolder+'cache.json', JSON.stringify(cache))
}
return [dz_id, dz_track, isrc]
}
async convertAlbum(dz, album_id){
if (!this.enabled) throw new Error("Spotify plugin not enabled")
let cachedAlbum
let cache
try {
cache = JSON.parse(fs.readFileSync(this.configFolder+'cache.json'))
} catch {
cache = {tracks: {}, albums: {}}
}
if (cache.albums[album_id]){
cachedAlbum = cache.albums[album_id]
} else {
try{
cachedAlbum = await this.sp.getAlbum(album_id)
} catch (e){
if (e.body.error.message === "invalid id") throw new InvalidID(`https://open.spotify.com/album/${album_id}`)
throw e
}
cachedAlbum = cachedAlbum.body
}
let dz_id = "0"
let dz_album = null
let upc = null
if (cachedAlbum.external_ids && cachedAlbum.external_ids.upc){
upc = cachedAlbum.external_ids.upc
try {
dz_album = await dz.api.get_album_by_UPC(upc)
} catch (e){
dz_album = null
}
if (!dz_album){
upc = ""+parseInt(upc)
try {
dz_album = await dz.api.get_album_by_UPC(upc)
} catch (e) {
dz_album = null
}
}
if (dz_album && dz_album.title && dz_album.id) dz_id = dz_album.id
}
cache.albums[album_id] = {id: dz_id, upc: upc}
fs.writeFileSync(this.configFolder+'cache.json', JSON.stringify(cache))
return dz_id
}
async convert(dz, downloadObject, settings, listener = null){
let cache
try {
cache = JSON.parse(fs.readFileSync(this.configFolder+'cache.json'))
} catch {
cache = {tracks: {}, albums: {}}
}
let conversion = 0
let conversionNext = 0
let collection = []
if (listener) listener.send("startConversion", downloadObject.uuid)
let q = queue(async (data) => {
let {track, pos} = data
if (downloadObject.cancel) return
let dz_id, trackAPI
if (cache.tracks[track.id]){
dz_id = cache.tracks[track.id].id
if (cache.tracks[track.id].isrc) trackAPI = await dz.api.get_track_by_ISRC(cache.tracks[track.id].isrc)
} else {
let isrc
try{
[dz_id, trackAPI, isrc] = await this.convertTrack(dz, "0", settings.fallbackSearch, track)
}catch (e){
console.warn(e.message)
}
cache.tracks[track.id] = {
id: dz_id,
isrc
}
}
let deezerTrack
if (String(dz_id) == "0"){
deezerTrack = {
SNG_ID: "0",
SNG_TITLE: track.name,
DURATION: 0,
MD5_ORIGIN: 0,
MEDIA_VERSION: 0,
FILESIZE: 0,
ALB_TITLE: track.album.name,
ALB_PICTURE: "",
ART_ID: 0,
ART_NAME: track.artists[0].name
}
} else {
deezerTrack = await dz.gw.get_track_with_fallback(dz_id)
}
if (trackAPI) deezerTrack._EXTRA_TRACK = trackAPI
deezerTrack.POSITION = pos
deezerTrack.SIZE = downloadObject.size
collection.push(deezerTrack)
conversionNext += (1 / downloadObject.size) * 100
if (Math.round(conversionNext) != conversion && Math.round(conversionNext) % 2 == 0){
conversion = Math.round(conversionNext)
if (listener) listener.send('updateQueue', {uuid: downloadObject.uuid, conversion})
}
}, settings.queueConcurrency)
downloadObject.conversion_data.forEach((track, pos) => {
q.push({track, pos: pos+1})
});
await q.drain()
downloadObject.collection.tracks_gw = collection
downloadObject.size = collection.length
downloadObject = new Collection(downloadObject.toDict())
if (listener) listener.send("finishConversion", downloadObject.getSlimmedDict())
fs.writeFileSync(this.configFolder+'cache.json', JSON.stringify(cache))
return downloadObject
}
_convertPlaylistStructure(spotifyPlaylist){
let cover
if (spotifyPlaylist.images.length) cover = spotifyPlaylist.images[0].url
else cover = null
let deezerPlaylist = {
checksum: spotifyPlaylist.snapshot_id,
collaborative: spotifyPlaylist.collaborative,
creation_date: "XXXX-00-00",
creator: {
id: spotifyPlaylist.owner.id,
name: spotifyPlaylist.owner.display_name,
tracklist: spotifyPlaylist.owner.href,
type: "user"
},
description: spotifyPlaylist.description,
duration: 0,
fans: spotifyPlaylist.followers ? spotifyPlaylist.followers.total : 0,
id: spotifyPlaylist.id,
is_loved_track: false,
link: spotifyPlaylist.external_urls.spotify,
nb_tracks: spotifyPlaylist.tracks.total,
picture: cover,
picture_small: cover || "https://e-cdns-images.dzcdn.net/images/cover/d41d8cd98f00b204e9800998ecf8427e/56x56-000000-80-0-0.jpg",
picture_medium: cover || "https://e-cdns-images.dzcdn.net/images/cover/d41d8cd98f00b204e9800998ecf8427e/250x250-000000-80-0-0.jpg",
picture_big: cover || "https://e-cdns-images.dzcdn.net/images/cover/d41d8cd98f00b204e9800998ecf8427e/500x500-000000-80-0-0.jpg",
picture_xl: cover || "https://e-cdns-images.dzcdn.net/images/cover/d41d8cd98f00b204e9800998ecf8427e/1000x1000-000000-80-0-0.jpg",
picture_thumbnail: cover || "https://e-cdns-images.dzcdn.net/images/cover/d41d8cd98f00b204e9800998ecf8427e/75x75-000000-80-0-0.jpg",
public: spotifyPlaylist.public,
share: spotifyPlaylist.external_urls.spotify,
title: spotifyPlaylist.name,
tracklist: spotifyPlaylist.tracks.href,
type: "playlist"
}
return deezerPlaylist
}
checkCredentials(){
if (this.credentials.clientId === "" || this.credentials.clientSecret === ""){
this.enabled = false
return
}
this.sp = new SpotifyWebApi(this.credentials)
this.sp.clientCredentialsGrant().then(
(creds)=>{
this.sp.setAccessToken(creds.body.access_token)
this.enabled = true
},
()=>{
this.enabled = false
this.sp = undefined
}
)
}
getCredentials(){
return this.credentials
}
setCredentials(newCredentials){
newCredentials.clientId = newCredentials.clientId.trim()
newCredentials.clientSecret = newCredentials.clientSecret.trim()
this.credentials = newCredentials
fs.writeFileSync(this.configFolder+'credentials.json', JSON.stringify(this.credentials))
this.checkCredentials()
}
}
module.exports = Spotify