Install
git clone https://github.com/panticz/spotifyripper.git cd spotifyripper ./spotifyripper.py
Spotify ripper
https://github.com/panticz/spotifyripper
#! /usr/bin/env python3
# Install the required Python modules
# pip install pulsectl pydub
import dbus
import os
import pprint
import pulsectl
import re
import requests
import subprocess
from dbus.mainloop.glib import DBusGMainLoop
from gi.repository import GLib
from pydub import AudioSegment
pre_subprocess = None
r = None
pre_album = ""
pre_artist = ""
pre_title = ""
pre_art_url = ""
pre_track_number = ""
pre_file_input = ""
pre_file_cover = ""
file_cover = ""
spotify_sink_index = 0
def get_spotify_sink_index():
with pulsectl.Pulse('spotify') as pulse:
for sink in pulse.sink_input_list():
# print("sink.name:" + sink.name)
# print("sink.corked:" + str(sink.corked))
if (sink.name == "Spotify") and (sink.corked == False):
return sink.index
return 0
def create_directory(path_album):
# print("path_album: " + path_album)
# re.sub("[^a-zA-Z]+", "", path_album)
re.sub("/", "-", path_album)
# print("path_album: " + path_album)
try:
os.makedirs(path_album, exist_ok=True)
except OSError:
print("Creation of the directory %s failed, use /tmp" % path_album)
return "/tmp"
return path_album
def download_cover(art_url, file_cover):
global r
# print("file_cover: " + file_cover)
# print("art_url: " + art_url)
if (art_url != ""):
# if art_url != pre_art_url:
r = requests.get(art_url, allow_redirects=True)
if r != None:
open(file_cover, 'wb').write(r.content)
def spotify_handler(*args):
global pre_album
global pre_artist
global pre_title
global pre_subprocess
global pre_art_url
global pre_track_number
global pre_file_input
global pre_file_cover
global r
global spotify_sink_index
metadata = args[1]["Metadata"]
# debug
# pprint.pprint(metadata)
artist = metadata["xesam:artist"][0]
# print("Artist: " + artist)
#albumArtist = metadata["xesam:albumArtist"][0]
# print("AArtist " + albumArtist)
title = metadata["xesam:title"]
album = metadata["xesam:album"]
track_number = metadata["xesam:trackNumber"]
art_url = metadata["mpris:artUrl"]
disc_number = int(metadata["xesam:discNumber"])
# if albumArtist != "":
# artist = albumArtist
if title != "":
title = title.replace("/", "-")
# workaround for art URL
art_url = art_url.replace("open.spotify.com", "i.scdn.co")
if title != pre_title:
print("Artist: " + artist)
print("Album: " + album)
print("Title: " + str(track_number) + " - " + title)
# print("Cover: " + art_url)
# print("Track: " + str(track_number))
print()
# create dir
path_base = os.path.expanduser("~/Downloads/spotifyripper")
if disc_number > 1:
disc_number_str = str(disc_number) + " "
else:
disc_number_str = ""
path_album = create_directory(path_base + "/" + artist + "/" + disc_number_str + album)
# print("path_album: " + path_album)
# record stream
if pre_subprocess != None:
pre_subprocess.terminate()
file_input = path_album + "/" + str(track_number) + " - " + artist + " - " + title + ".wav"
if spotify_sink_index == 0:
spotify_sink_index = get_spotify_sink_index()
print("spotify_sink_index: " + str(spotify_sink_index))
if (artist != "") or (album != ""):
pre_subprocess = subprocess.Popen(["parec", "--monitor-stream=" + str(spotify_sink_index), "--file-format=wav", file_input])
# convert previous file
if os.path.isfile(pre_file_input):
pre_file_output = pre_file_input.replace(".wav", ".mp3")
sound = AudioSegment.from_wav(pre_file_input)
sound.export(pre_file_output, format="mp3", bitrate="160k", cover=pre_file_cover, tags={
"album": pre_album,
"artist": pre_artist,
"title": pre_title,
"track": int(pre_track_number)
}
)
pre_file_output_size = os.stat(pre_file_output).st_size
if pre_file_output_size < 1048576:
print('\033[33m' + "Warning: small file " + pre_file_output + " \033[0m\n")
if pre_file_output_size > 10485760:
print('\033[33m' + "Warning: large file " + pre_file_output + " \033[0m\n")
# print("DELETE " + pre_file_cover)
os.remove(pre_file_cover)
# print("DELETE " + pre_file_input)
os.remove(pre_file_input)
# download cover
file_cover = file_input.replace(".wav", ".jpg")
download_cover(art_url, file_cover)
if art_url != "":
pre_art_url = art_url
if file_cover != "":
pre_file_cover = file_cover
pre_file_input = file_input
if album != "":
pre_album = album
if artist != "":
pre_artist = artist
if title != "":
pre_title = title
if track_number != "":
pre_track_number = track_number
spotify_sink_index = get_spotify_sink_index()
DBusGMainLoop(set_as_default=True)
session_bus = dbus.SessionBus()
session_bus.add_signal_receiver(spotify_handler, 'PropertiesChanged', None, 'org.mpris.MediaPlayer2.spotify', '/org/mpris/MediaPlayer2')
loop = GLib.MainLoop()
loop.run()
>
Record stream
SPOTIFY=$(pacmd list-sink-inputs | while read line; do [[ -n $(echo $line | grep "index:") ]] && index=$line [[ -n $(echo $line | grep Spotify) ]] && echo $index && exit done | cut -d: -f2) echo ${SPOTIFY} # muted pactl load-module module-null-sink 'sink_name=spotify' pactl move-sink-input ${SPOTIFY} spotify parec -d spotify.monitor | oggenc -b 320 -o /tmp/spotify.ogg --raw - parec --verbose --monitor-stream=${SPOTIFY} | oggenc -b 320 -o /tmp/spotify.ogg --raw - # v2 # unmuted #parec --verbose --monitor-stream=${SPOTIFY} | oggencode -b 320 -o /tmp/spotify.ogg --raw - parec --verbose --monitor-stream=${SPOTIFY} --file-format=wav /tmp/recording.wav #parec --verbose --monitor-stream=${SPOTIFY} | lame -r --quiet -q 3 --lowpass 17 --abr 192 - /tmp/spotify.mp3 parec -d record-n-play.monitor # cleanup / restart pulseaudio # systemctl --user restart pulseaudio.service pulseaudio -k && pulseaudio --start
Links
https://amish.naidu.dev/blog/dbus/
https://ccoffey.ie/?p=53
https://programtalk.com/vs2/python/14193/spotify-music-downloader/spotify_downloader.py/#
https://pypi.org/project/pulsectl/#usage
http://panticz.de/pulseaudio