2021-05-28 15:22:07 +00:00
|
|
|
const Plugin = require('./plugin.js')
|
|
|
|
const { getConfigFolder } = require('../utils/localpaths.js')
|
|
|
|
const {
|
|
|
|
generateTrackItem,
|
|
|
|
generateAlbumItem,
|
|
|
|
TrackNotOnDeezer,
|
2021-06-05 16:20:22 +00:00
|
|
|
AlbumNotOnDeezer,
|
|
|
|
InvalidID
|
2021-05-28 15:22:07 +00:00
|
|
|
} = require('../itemgen.js')
|
2021-05-29 08:43:09 +00:00
|
|
|
const { Convertable, Collection } = require('../types/DownloadObjects.js')
|
2021-05-28 15:22:07 +00:00
|
|
|
const { sep } = require('path')
|
|
|
|
const fs = require('fs')
|
|
|
|
const SpotifyWebApi = require('spotify-web-api-node')
|
|
|
|
const got = require('got')
|
2021-05-29 08:43:09 +00:00
|
|
|
const { queue } = require('async')
|
2021-05-28 15:22:07 +00:00
|
|
|
|
|
|
|
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
|
|
|
|
}
|
|
|
|
|
2021-05-29 10:03:32 +00:00
|
|
|
setup(){
|
2021-05-28 15:22:07 +00:00
|
|
|
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'))
|
2021-05-29 10:03:32 +00:00
|
|
|
this.checkCredentials()
|
|
|
|
return this
|
2021-05-28 15:22:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
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]
|
|
|
|
}
|
|
|
|
|
2021-05-29 08:43:09 +00:00
|
|
|
async generateDownloadObject(dz, link, bitrate){
|
2021-05-28 15:22:07 +00:00
|
|
|
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){
|
2021-05-29 08:43:09 +00:00
|
|
|
let [track_id, trackAPI] = await this.convertTrack(dz, link_id)
|
2021-05-28 15:22:07 +00:00
|
|
|
|
|
|
|
if (track_id !== "0"){
|
|
|
|
return generateTrackItem(dz, track_id, bitrate, trackAPI)
|
|
|
|
} else {
|
2021-06-05 16:20:22 +00:00
|
|
|
throw new TrackNotOnDeezer(`https://open.spotify.com/track/${link_id}`)
|
2021-05-28 15:22:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
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 {
|
2021-06-05 16:20:22 +00:00
|
|
|
throw new AlbumNotOnDeezer(`https://open.spotify.com/album/${link_id}`)
|
2021-05-28 15:22:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
async generatePlaylistItem(dz, link_id, bitrate){
|
2021-05-29 08:43:09 +00:00
|
|
|
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
|
|
|
|
})
|
2021-05-28 15:22:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
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){
|
2021-05-29 08:43:09 +00:00
|
|
|
try {
|
|
|
|
cache = JSON.parse(fs.readFileSync(this.configFolder+'cache.json'))
|
|
|
|
} catch {
|
|
|
|
cache = {tracks: {}, albums: {}}
|
|
|
|
}
|
2021-05-28 15:22:07 +00:00
|
|
|
shouldSaveCache = true
|
2021-05-29 08:43:09 +00:00
|
|
|
if (cache.tracks[track_id]){
|
|
|
|
cachedTrack = cache.tracks[track_id]
|
|
|
|
} else {
|
2021-06-05 16:20:22 +00:00
|
|
|
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
|
|
|
|
}
|
2021-05-29 08:43:09 +00:00
|
|
|
cachedTrack = cachedTrack.body
|
|
|
|
}
|
2021-05-28 15:22:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
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}
|
2021-05-29 08:43:09 +00:00
|
|
|
fs.writeFileSync(this.configFolder+'cache.json', JSON.stringify(cache))
|
2021-05-28 15:22:07 +00:00
|
|
|
}
|
|
|
|
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
|
2021-05-29 08:43:09 +00:00
|
|
|
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 {
|
2021-06-05 16:20:22 +00:00
|
|
|
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
|
|
|
|
}
|
2021-05-28 15:22:07 +00:00
|
|
|
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
|
|
|
|
}
|
|
|
|
|
2021-05-29 08:43:09 +00:00
|
|
|
cache.albums[album_id] = {id: dz_id, upc: upc}
|
|
|
|
fs.writeFileSync(this.configFolder+'cache.json', JSON.stringify(cache))
|
2021-05-28 15:22:07 +00:00
|
|
|
return dz_id
|
|
|
|
}
|
|
|
|
|
2021-05-29 08:43:09 +00:00
|
|
|
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 = []
|
2021-05-29 08:53:12 +00:00
|
|
|
if (listener) listener.send("startConversion", downloadObject.uuid)
|
2021-05-29 08:43:09 +00:00
|
|
|
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
|
2021-06-05 16:20:22 +00:00
|
|
|
try{
|
|
|
|
[dz_id, trackAPI, isrc] = await this.convertTrack(dz, "0", settings.fallbackSearch, track)
|
|
|
|
}catch (e){
|
|
|
|
console.warn(e.message)
|
|
|
|
}
|
|
|
|
|
2021-05-29 08:43:09 +00:00
|
|
|
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())
|
2021-06-01 18:11:02 +00:00
|
|
|
if (listener) listener.send("finishConversion", downloadObject.getSlimmedDict())
|
2021-05-29 08:43:09 +00:00
|
|
|
|
|
|
|
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
|
|
|
|
}
|
|
|
|
|
2021-05-29 10:03:32 +00:00
|
|
|
checkCredentials(){
|
2021-05-28 15:22:07 +00:00
|
|
|
if (this.credentials.clientId === "" || this.credentials.clientSecret === ""){
|
|
|
|
this.enabled = false
|
|
|
|
return
|
|
|
|
}
|
|
|
|
this.sp = new SpotifyWebApi(this.credentials)
|
2021-05-29 10:03:32 +00:00
|
|
|
this.sp.clientCredentialsGrant().then(
|
|
|
|
(creds)=>{
|
|
|
|
this.sp.setAccessToken(creds.body.access_token)
|
|
|
|
this.enabled = true
|
|
|
|
},
|
|
|
|
()=>{
|
|
|
|
this.enabled = false
|
|
|
|
this.sp = undefined
|
|
|
|
}
|
|
|
|
)
|
2021-05-28 15:22:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
getCredentials(){
|
|
|
|
return this.credentials
|
|
|
|
}
|
|
|
|
|
2021-05-29 10:03:32 +00:00
|
|
|
setCredentials(newCredentials){
|
2021-05-28 15:22:07 +00:00
|
|
|
newCredentials.clientId = newCredentials.clientId.trim()
|
|
|
|
newCredentials.clientSecret = newCredentials.clientSecret.trim()
|
|
|
|
|
|
|
|
this.credentials = newCredentials
|
|
|
|
fs.writeFileSync(this.configFolder+'credentials.json', JSON.stringify(this.credentials))
|
2021-05-29 10:03:32 +00:00
|
|
|
this.checkCredentials()
|
2021-05-28 15:22:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
module.exports = Spotify
|