2020-08-23 11:23:09 -04:00
|
|
|
import os
|
|
|
|
|
import json
|
|
|
|
|
import random
|
|
|
|
|
import requests
|
2020-08-15 22:22:39 -04:00
|
|
|
|
2020-08-23 11:23:09 -04:00
|
|
|
from base64 import b64encode
|
2020-08-23 11:18:07 -04:00
|
|
|
from dotenv import load_dotenv, find_dotenv
|
2021-07-09 20:39:46 +03:00
|
|
|
from flask import Flask, Response, jsonify, render_template, templating
|
2020-08-15 22:22:39 -04:00
|
|
|
|
2020-08-23 11:23:09 -04:00
|
|
|
load_dotenv(find_dotenv())
|
2020-08-23 11:18:07 -04:00
|
|
|
|
2020-09-01 19:13:53 -04:00
|
|
|
# Spotify scopes:
|
2020-08-23 11:23:09 -04:00
|
|
|
# user-read-currently-playing
|
|
|
|
|
# user-read-recently-played
|
2021-01-28 19:27:25 +01:00
|
|
|
PLACEHOLDER_IMAGE = "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
|
2020-08-15 22:22:39 -04:00
|
|
|
SPOTIFY_CLIENT_ID = os.getenv("SPOTIFY_CLIENT_ID")
|
|
|
|
|
SPOTIFY_SECRET_ID = os.getenv("SPOTIFY_SECRET_ID")
|
|
|
|
|
SPOTIFY_REFRESH_TOKEN = os.getenv("SPOTIFY_REFRESH_TOKEN")
|
2021-01-28 19:27:25 +01:00
|
|
|
|
2021-07-09 20:44:19 +03:00
|
|
|
#########################
|
|
|
|
|
# change theme light/dark
|
|
|
|
|
|
2021-07-09 20:39:46 +03:00
|
|
|
TEMPLATE_THEME = "dark"
|
|
|
|
|
|
2020-09-01 19:13:53 -04:00
|
|
|
REFRESH_TOKEN_URL = "https://accounts.spotify.com/api/token"
|
|
|
|
|
NOW_PLAYING_URL = "https://api.spotify.com/v1/me/player/currently-playing"
|
|
|
|
|
RECENTLY_PLAYING_URL = (
|
|
|
|
|
"https://api.spotify.com/v1/me/player/recently-played?limit=10"
|
|
|
|
|
)
|
2020-08-15 22:22:39 -04:00
|
|
|
|
|
|
|
|
app = Flask(__name__)
|
|
|
|
|
|
2020-09-01 19:13:53 -04:00
|
|
|
|
2020-08-15 22:22:39 -04:00
|
|
|
def getAuth():
|
2020-09-01 19:13:53 -04:00
|
|
|
return b64encode(f"{SPOTIFY_CLIENT_ID}:{SPOTIFY_SECRET_ID}".encode()).decode(
|
|
|
|
|
"ascii"
|
|
|
|
|
)
|
|
|
|
|
|
2020-08-15 22:22:39 -04:00
|
|
|
|
|
|
|
|
def refreshToken():
|
|
|
|
|
data = {
|
|
|
|
|
"grant_type": "refresh_token",
|
|
|
|
|
"refresh_token": SPOTIFY_REFRESH_TOKEN,
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
headers = {"Authorization": "Basic {}".format(getAuth())}
|
2020-09-01 19:13:53 -04:00
|
|
|
response = requests.post(REFRESH_TOKEN_URL, data=data, headers=headers)
|
2020-12-17 18:39:39 -05:00
|
|
|
|
2020-09-11 17:18:40 -04:00
|
|
|
try:
|
|
|
|
|
return response.json()["access_token"]
|
|
|
|
|
except KeyError:
|
|
|
|
|
print(json.dumps(response.json()))
|
|
|
|
|
print("\n---\n")
|
|
|
|
|
raise KeyError(str(response.json()))
|
2020-08-15 22:22:39 -04:00
|
|
|
|
2020-09-01 19:13:53 -04:00
|
|
|
|
2020-08-15 22:22:39 -04:00
|
|
|
def recentlyPlayed():
|
|
|
|
|
token = refreshToken()
|
|
|
|
|
headers = {"Authorization": f"Bearer {token}"}
|
2020-09-01 19:13:53 -04:00
|
|
|
response = requests.get(RECENTLY_PLAYING_URL, headers=headers)
|
2020-08-15 22:22:39 -04:00
|
|
|
|
|
|
|
|
if response.status_code == 204:
|
|
|
|
|
return {}
|
|
|
|
|
return response.json()
|
|
|
|
|
|
2020-09-01 19:13:53 -04:00
|
|
|
|
2020-08-15 22:22:39 -04:00
|
|
|
def nowPlaying():
|
|
|
|
|
token = refreshToken()
|
|
|
|
|
headers = {"Authorization": f"Bearer {token}"}
|
2020-09-01 19:13:53 -04:00
|
|
|
response = requests.get(NOW_PLAYING_URL, headers=headers)
|
2020-08-15 22:22:39 -04:00
|
|
|
|
|
|
|
|
if response.status_code == 204:
|
|
|
|
|
return {}
|
|
|
|
|
return response.json()
|
|
|
|
|
|
2020-09-01 19:13:53 -04:00
|
|
|
|
2020-08-15 22:22:39 -04:00
|
|
|
def barGen(barCount):
|
|
|
|
|
barCSS = ""
|
|
|
|
|
left = 1
|
|
|
|
|
for i in range(1, barCount + 1):
|
|
|
|
|
anim = random.randint(1000, 1350)
|
2020-09-01 19:13:53 -04:00
|
|
|
barCSS += (
|
|
|
|
|
".bar:nth-child({}) {{ left: {}px; animation-duration: {}ms; }}".format(
|
|
|
|
|
i, left, anim
|
|
|
|
|
)
|
2020-08-15 22:22:39 -04:00
|
|
|
)
|
|
|
|
|
left += 4
|
|
|
|
|
return barCSS
|
|
|
|
|
|
2021-07-09 20:44:19 +03:00
|
|
|
def getTemplate(theme):
|
2021-07-09 20:39:46 +03:00
|
|
|
return{
|
|
|
|
|
'dark':'spotify-dark.html.j2',
|
|
|
|
|
'light':'spotify.html.js'
|
|
|
|
|
}[theme.lower()]
|
|
|
|
|
|
2020-09-01 19:13:53 -04:00
|
|
|
|
2020-08-15 22:22:39 -04:00
|
|
|
def loadImageB64(url):
|
|
|
|
|
resposne = requests.get(url)
|
|
|
|
|
return b64encode(resposne.content).decode("ascii")
|
|
|
|
|
|
2020-09-01 19:13:53 -04:00
|
|
|
|
2020-08-15 22:22:39 -04:00
|
|
|
def makeSVG(data):
|
2020-08-23 13:08:24 -04:00
|
|
|
barCount = 84
|
2020-08-15 22:22:39 -04:00
|
|
|
contentBar = "".join(["<div class='bar'></div>" for i in range(barCount)])
|
|
|
|
|
barCSS = barGen(barCount)
|
|
|
|
|
|
2020-12-17 18:39:39 -05:00
|
|
|
if data == {} or data["item"] == "None" or data["item"] is None:
|
2020-09-01 19:13:53 -04:00
|
|
|
# contentBar = "" #Shows/Hides the EQ bar if no song is currently playing
|
|
|
|
|
currentStatus = "Was playing:"
|
2020-08-23 11:23:09 -04:00
|
|
|
recentPlays = recentlyPlayed()
|
|
|
|
|
recentPlaysLength = len(recentPlays["items"])
|
|
|
|
|
itemIndex = random.randint(0, recentPlaysLength - 1)
|
|
|
|
|
item = recentPlays["items"][itemIndex]["track"]
|
2020-08-15 22:22:39 -04:00
|
|
|
else:
|
|
|
|
|
item = data["item"]
|
2020-08-25 19:21:33 -04:00
|
|
|
currentStatus = "Vibing to:"
|
2021-01-28 19:15:14 +01:00
|
|
|
|
|
|
|
|
if item["album"]["images"] == []:
|
|
|
|
|
image = PLACEHOLDER_IMAGE
|
|
|
|
|
else :
|
|
|
|
|
image = loadImageB64(item["album"]["images"][1]["url"])
|
|
|
|
|
|
2020-08-15 22:22:39 -04:00
|
|
|
artistName = item["artists"][0]["name"].replace("&", "&")
|
|
|
|
|
songName = item["name"].replace("&", "&")
|
|
|
|
|
|
|
|
|
|
dataDict = {
|
2020-08-23 11:23:09 -04:00
|
|
|
"contentBar": contentBar,
|
|
|
|
|
"barCSS": barCSS,
|
|
|
|
|
"artistName": artistName,
|
|
|
|
|
"songName": songName,
|
|
|
|
|
"image": image,
|
2020-09-01 19:13:53 -04:00
|
|
|
"status": currentStatus,
|
2020-08-15 22:22:39 -04:00
|
|
|
}
|
|
|
|
|
|
2021-07-09 20:44:19 +03:00
|
|
|
return render_template(getTemplate(TEMPLATE_THEME), **dataDict)
|
2020-08-15 22:22:39 -04:00
|
|
|
|
2020-09-01 19:13:53 -04:00
|
|
|
|
2020-08-15 22:22:39 -04:00
|
|
|
@app.route("/", defaults={"path": ""})
|
|
|
|
|
@app.route("/<path:path>")
|
|
|
|
|
def catch_all(path):
|
|
|
|
|
data = nowPlaying()
|
|
|
|
|
svg = makeSVG(data)
|
|
|
|
|
|
|
|
|
|
resp = Response(svg, mimetype="image/svg+xml")
|
|
|
|
|
resp.headers["Cache-Control"] = "s-maxage=1"
|
|
|
|
|
|
|
|
|
|
return resp
|
|
|
|
|
|
2020-09-01 19:13:53 -04:00
|
|
|
|
2020-08-15 22:22:39 -04:00
|
|
|
if __name__ == "__main__":
|
|
|
|
|
app.run(debug=True)
|