@ -0,0 +1,4 @@ | |||||
.vercel | |||||
.env | |||||
node_modules | |||||
.DS_STORE |
@ -1 +1,3 @@ | |||||
[Spotify](https://novatorem.vercel.app/api/spotify-playing) | |||||
<a href="https://novatorem.vercel.app/now-playing?open"> | |||||
<img src="https://novatorem.vercel.app/now-playing" width="256" height="64" alt="Now Playing"> | |||||
</a> |
@ -1,11 +0,0 @@ | |||||
from http.server import BaseHTTPRequestHandler | |||||
from datetime import datetime | |||||
class handler(BaseHTTPRequestHandler): | |||||
def do_GET(self): | |||||
self.send_response(200) | |||||
self.send_header('Content-type', 'text/plain') | |||||
self.end_headers() | |||||
self.wfile.write(str(datetime.now().strftime('%Y-%m-%d %H:%M:%S')).encode()) | |||||
return |
@ -0,0 +1,44 @@ | |||||
import { NowRequest, NowResponse } from "@vercel/node"; | |||||
import { renderToString } from "react-dom/server"; | |||||
import { decode } from "querystring"; | |||||
import { Player } from "../components/NowPlaying"; | |||||
import { nowPlaying } from "../utils/spotify"; | |||||
export default async function (req: NowRequest, res: NowResponse) { | |||||
const { | |||||
item = {}, | |||||
is_playing: isPlaying = false, | |||||
progress_ms: progress = 0, | |||||
} = await nowPlaying(); | |||||
const params = decode(req.url.split("?")[1]) as any; | |||||
if (params && typeof params.open !== "undefined") { | |||||
if (item && item.external_urls) { | |||||
res.writeHead(302, { | |||||
Location: item.external_urls.spotify, | |||||
}); | |||||
return res.end(); | |||||
} | |||||
return res.status(200).end(); | |||||
} | |||||
res.setHeader("Content-Type", "image/svg+xml"); | |||||
res.setHeader("Cache-Control", "s-maxage=1, stale-while-revalidate"); | |||||
const { duration_ms: duration, name: track } = item; | |||||
const { images = [] } = item.album || {}; | |||||
const cover = images[images.length - 1]?.url; | |||||
let coverImg = null; | |||||
if (cover) { | |||||
const buff = await (await fetch(cover)).arrayBuffer(); | |||||
coverImg = `data:image/jpeg;base64,${Buffer.from(buff).toString("base64")}`; | |||||
} | |||||
const artist = (item.artists || []).map(({ name }) => name).join(", "); | |||||
const text = renderToString( | |||||
Player({ cover: coverImg, artist, track, isPlaying, progress, duration }) | |||||
); | |||||
return res.status(200).send(text); | |||||
} |
@ -1,3 +0,0 @@ | |||||
flask==1.1.2 | |||||
requests==2.24.0 | |||||
python-dotenv==0.14.0 |
@ -1,216 +0,0 @@ | |||||
from flask import Flask, Response, jsonify | |||||
from base64 import b64encode | |||||
from dotenv import load_dotenv, find_dotenv | |||||
load_dotenv(find_dotenv()) | |||||
import requests | |||||
import json | |||||
import os | |||||
import random | |||||
""" | |||||
Inspired from https://github.com/natemoo-re | |||||
""" | |||||
print("Starting Server") | |||||
SPOTIFY_CLIENT_ID = os.getenv("SPOTIFY_CLIENT_ID") | |||||
SPOTIFY_SECRET_ID = os.getenv("SPOTIFY_SECRET_ID") | |||||
SPOTIFY_REFRESH_TOKEN = os.getenv("SPOTIFY_REFRESH_TOKEN") | |||||
SPOTIFY_URL_REFRESH_TOKEN = "https://accounts.spotify.com/api/token" | |||||
SPOTIFY_URL_NOW_PLAYING = "https://api.spotify.com/v1/me/player/currently-playing" | |||||
LATEST_PLAY = None | |||||
app = Flask(__name__) | |||||
def get_authorization(): | |||||
return b64encode(f"{SPOTIFY_CLIENT_ID}:{SPOTIFY_SECRET_ID}".encode()).decode("ascii") | |||||
def refresh_token(): | |||||
data = { | |||||
"grant_type": "refresh_token", | |||||
"refresh_token": SPOTIFY_REFRESH_TOKEN, | |||||
} | |||||
headers = {"Authorization": "Basic {}".format(get_authorization())} | |||||
response = requests.post(SPOTIFY_URL_REFRESH_TOKEN, data=data, headers=headers) | |||||
repsonse_json = response.json() | |||||
return repsonse_json["access_token"] | |||||
def get_now_playing(): | |||||
token = refresh_token() | |||||
headers = {"Authorization": f"Bearer {token}"} | |||||
response = requests.get(SPOTIFY_URL_NOW_PLAYING, headers=headers) | |||||
if response.status_code == 204: | |||||
return {} | |||||
repsonse_json = response.json() | |||||
return repsonse_json | |||||
def get_svg_template(): | |||||
css_bar = "" | |||||
left = 1 | |||||
for i in range(1, 76): | |||||
anim = random.randint(350, 500) | |||||
css_bar += ".bar:nth-child({}) {{{{ left: {}px; animation-duration: {}ms; }}}}".format( | |||||
i, left, anim | |||||
) | |||||
left += 4 | |||||
svg = ( | |||||
""" | |||||
<svg width="320" height="445" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink"> | |||||
<foreignObject width="320" height="445"> | |||||
<div xmlns="http://www.w3.org/1999/xhtml" class="container"> | |||||
<style> | |||||
div {{font-family: -apple-system,BlinkMacSystemFont,Segoe UI,Helvetica,Arial,sans-serif,Apple Color Emoji,Segoe UI Emoji;}} | |||||
.container {{background-color: #121212; border-radius: 10px; padding: 10px 10px}} | |||||
.playing {{ font-weight: bold; color: #53b14f; text-align: center; display: flex; justify-content: center; align-items: center;}} | |||||
.not-play {{color: #ff1616;}} | |||||
.artist {{ font-weight: bold; font-size: 20px; color: #fff; text-align: center; margin-top: 5px; }} | |||||
.song {{ font-size: 16px; color: #b3b3b3; text-align: center; margin-top: 5px; margin-bottom: 15px; }} | |||||
.logo {{ margin-left: 5px; margin-top: 5px; }} | |||||
.cover {{ border-radius: 5px; margin-top: 9px; }} | |||||
#bars {{ | |||||
height: 30px; | |||||
margin: -20px 0 0 0px; | |||||
position: absolute; | |||||
width: 40px; | |||||
}} | |||||
.bar {{ | |||||
background: #53b14f; | |||||
bottom: 1px; | |||||
height: 3px; | |||||
position: absolute; | |||||
width: 3px; | |||||
animation: sound 0ms -800ms linear infinite alternate; | |||||
}} | |||||
@keyframes sound {{ | |||||
0% {{ | |||||
opacity: .35; | |||||
height: 3px; | |||||
}} | |||||
100% {{ | |||||
opacity: 1; | |||||
height: 28px; | |||||
}} | |||||
}} | |||||
""" | |||||
+ css_bar | |||||
+ """ | |||||
</style> | |||||
{} | |||||
</div> | |||||
</foreignObject> | |||||
</svg> | |||||
""" | |||||
) | |||||
return svg | |||||
def load_image_b64(url): | |||||
resposne = requests.get(url) | |||||
return b64encode(resposne.content).decode("ascii") | |||||
def make_svg(data): | |||||
global LATEST_PLAY | |||||
template = get_svg_template() | |||||
text = "Now playing" | |||||
content_bar = "".join(["<div class='bar'></div>" for i in range(75)]) | |||||
if data == {} and LATEST_PLAY is not None: | |||||
data = LATEST_PLAY | |||||
text = "Latest play" | |||||
content_bar = "" | |||||
elif data == {}: | |||||
content = """ | |||||
<div class="playing not-play">Nothing playing on Spotify</div> | |||||
""" | |||||
return template.format(content) | |||||
content = """ | |||||
<div class="playing">{} on <img class="logo" src="" /></div> | |||||
<div class="artist">{}</div> | |||||
<div class="song">{}</div> | |||||
<div id='bars'> | |||||
{} | |||||
</div> | |||||
<a href="{}" target="_BLANK"> | |||||
<center> | |||||
<img src="data:image/png;base64, {}" width="300" height="300" class="cover"/> | |||||
</center> | |||||
</a> | |||||
""" | |||||
item = data["item"] | |||||
""" | |||||
print(json.dumps(item)) | |||||
print(item["artists"][0]["name"]) | |||||
print(item["external_urls"]["spotify"]) | |||||
print(item["album"]["images"][0]["url"]) | |||||
""" | |||||
img = load_image_b64(item["album"]["images"][1]["url"]) | |||||
artist_name = item["artists"][0]["name"].replace("&", "&") | |||||
song_name = item["name"].replace("&", "&") | |||||
content_rendered = content.format( | |||||
text, | |||||
artist_name, | |||||
song_name, | |||||
content_bar, | |||||
item["external_urls"]["spotify"], | |||||
img, | |||||
) | |||||
return template.format(content_rendered) | |||||
@app.route("/", defaults={"path": ""}) | |||||
@app.route("/<path:path>") | |||||
def catch_all(path): | |||||
global LATEST_PLAY | |||||
# TODO: caching | |||||
data = get_now_playing() | |||||
svg = make_svg(data) | |||||
# cache lastest data | |||||
if data != {}: | |||||
LATEST_PLAY = data | |||||
resp = Response(svg, mimetype="image/svg+xml") | |||||
resp.headers["Cache-Control"] = "s-maxage=1" | |||||
return resp | |||||
if __name__ == "__main__": | |||||
app.run(debug=True) |
@ -0,0 +1,162 @@ | |||||
import React from "react"; | |||||
import ReadmeImg from "./ReadmeImg"; | |||||
import Text from "./Text"; | |||||
export interface Props { | |||||
cover?: string; | |||||
track: string; | |||||
artist: string; | |||||
progress: number; | |||||
duration: number; | |||||
isPlaying: boolean; | |||||
} | |||||
export const Player: React.FC<Props> = ({ | |||||
cover, | |||||
track, | |||||
artist, | |||||
progress, | |||||
duration, | |||||
isPlaying, | |||||
}) => { | |||||
return ( | |||||
<ReadmeImg width="256" height="64"> | |||||
<style> | |||||
{` | |||||
.paused { | |||||
animation-play-state: paused !important; | |||||
background: #e1e4e8 !important; | |||||
} | |||||
img:not([src]) { | |||||
content: url(""); | |||||
border-radius: 6px; | |||||
background: #FFF; | |||||
border: 1px solid #e1e4e8; | |||||
} | |||||
p { | |||||
display: block; | |||||
opacity: 0; | |||||
} | |||||
.progress-bar { | |||||
position: relative; | |||||
width: 100%; | |||||
height: 4px; | |||||
margin: -1px; | |||||
border: 1px solid #e1e4e8; | |||||
border-radius: 4px; | |||||
overflow: hidden; | |||||
padding: 2px; | |||||
z-index: 0; | |||||
} | |||||
#progress { | |||||
position: absolute; | |||||
top: -1px; | |||||
left: 0; | |||||
width: 100%; | |||||
height: 6px; | |||||
transform-origin: left center; | |||||
background-color: #24292e; | |||||
animation: progress ${duration}ms linear; | |||||
animation-delay: -${progress}ms; | |||||
} | |||||
.progress-bar, | |||||
#track, | |||||
#artist, | |||||
#cover { | |||||
opacity: 0; | |||||
animation: appear 300ms ease-out forwards; | |||||
} | |||||
#track { | |||||
animation-delay: 400ms; | |||||
} | |||||
#artist { | |||||
animation-delay: 500ms; | |||||
} | |||||
.progress-bar { | |||||
animation-delay: 550ms; | |||||
margin-top: 4px; | |||||
} | |||||
#cover { | |||||
animation-name: cover-appear; | |||||
animation-delay: 300ms; | |||||
box-shadow: 0 1px 3px rgba(0,0,0,0.1), 0 3px 10px rgba(0,0,0,0.05); | |||||
} | |||||
#cover:not([src]) { | |||||
box-shadow: none; | |||||
} | |||||
@keyframes cover-appear { | |||||
from { | |||||
opacity: 0; | |||||
transform: scale(0.8); | |||||
} | |||||
to { | |||||
opacity: 1; | |||||
transform: scale(1); | |||||
} | |||||
} | |||||
@keyframes appear { | |||||
from { | |||||
opacity: 0; | |||||
transform: translateX(-8px); | |||||
} | |||||
to { | |||||
opacity: 1; | |||||
transform: translateX(0); | |||||
} | |||||
} | |||||
@keyframes progress { | |||||
from { | |||||
transform: scaleX(0) | |||||
} | |||||
to { | |||||
transform: scaleX(1) | |||||
} | |||||
} | |||||
`} | |||||
</style> | |||||
<div | |||||
className={isPlaying ? "disabled" : ""} | |||||
style={{ | |||||
display: "flex", | |||||
alignItems: "center", | |||||
paddingTop: 8, | |||||
paddingLeft: 4, | |||||
}} | |||||
> | |||||
<img id="cover" src={cover ?? null} width="48" height="48" /> | |||||
<div | |||||
style={{ | |||||
display: "flex", | |||||
flex: 1, | |||||
flexDirection: "column", | |||||
marginTop: -4, | |||||
marginLeft: 8, | |||||
}} | |||||
> | |||||
<Text id="track" weight="bold"> | |||||
{`${track ?? ""} `.trim()} | |||||
</Text> | |||||
<Text id="artist" color={!track ? "gray" : undefined}> | |||||
{artist || "Nothing playing..."} | |||||
</Text> | |||||
{track && ( | |||||
<div className="progress-bar"> | |||||
<div id="progress" className={!isPlaying ? "paused" : ""} /> | |||||
</div> | |||||
)} | |||||
</div> | |||||
</div> | |||||
</ReadmeImg> | |||||
); | |||||
}; |
@ -0,0 +1,27 @@ | |||||
import React from "react"; | |||||
const ReadmeImg = ({ width, height, children }) => { | |||||
return ( | |||||
<svg | |||||
fill="none" | |||||
width={width} | |||||
height={height} | |||||
viewBox={`0 0 ${width} ${height}`} | |||||
xmlns="http://www.w3.org/2000/svg" | |||||
> | |||||
<foreignObject width={width} height={height}> | |||||
<div {...{ xmlns: "http://www.w3.org/1999/xhtml" }}> | |||||
<style>{` | |||||
* { | |||||
margin: 0; | |||||
box-sizing: border-box; | |||||
} | |||||
`}</style> | |||||
{children} | |||||
</div> | |||||
</foreignObject> | |||||
</svg> | |||||
); | |||||
}; | |||||
export default ReadmeImg; |
@ -0,0 +1,51 @@ | |||||
import React from "react"; | |||||
const sizes = { | |||||
default: 14, | |||||
small: 12, | |||||
}; | |||||
const colors = { | |||||
default: "#24292e", | |||||
"gray-light": "#e1e4e8", | |||||
gray: "#586069", | |||||
"gray-dark": "#24292e", | |||||
}; | |||||
const families = { | |||||
default: | |||||
"-apple-system,BlinkMacSystemFont,Segoe UI,Helvetica,Arial,sans-serif,Apple Color Emoji,Segoe UI Emoji", | |||||
mono: "SFMono-Regular,Consolas,Liberation Mono,Menlo,monospace", | |||||
}; | |||||
const weights = { | |||||
default: 400, | |||||
bold: 600, | |||||
}; | |||||
const Text: React.FC<any> = ({ | |||||
children = "", | |||||
weight = "default", | |||||
family = "default", | |||||
color = "default", | |||||
size = "default", | |||||
...props | |||||
}) => { | |||||
return ( | |||||
<p | |||||
style={{ | |||||
whiteSpace: "pre", | |||||
fontSize: `${sizes[size]}px`, | |||||
lineHeight: 1.5, | |||||
fontFamily: families[family], | |||||
color: colors[color], | |||||
fontWeight: weights[weight], | |||||
}} | |||||
{...props} | |||||
> | |||||
{children} | |||||
</p> | |||||
); | |||||
}; | |||||
export default Text; |
@ -0,0 +1,43 @@ | |||||
{ | |||||
"name": "novatorem", | |||||
"private": true, | |||||
"version": "1.0.0", | |||||
"description": "Github profile", | |||||
"main": "index.js", | |||||
"dependencies": { | |||||
"faunadb": "^2.14.2", | |||||
"isomorphic-unfetch": "^3.0.0", | |||||
"querystring": "^0.2.0", | |||||
"react": "^16.13.1", | |||||
"react-dom": "^16.13.1", | |||||
"typescript": "^3.9.6" | |||||
}, | |||||
"devDependencies": { | |||||
"@types/node": "^14.0.22", | |||||
"@types/react": "^16.9.42", | |||||
"@types/react-dom": "^16.9.8", | |||||
"@vercel/node": "^1.7.2", | |||||
"husky": "^4.2.5", | |||||
"lint-staged": "^10.2.11", | |||||
"prettier": "^2.0.5" | |||||
}, | |||||
"repository": { | |||||
"type": "git", | |||||
"url": "git+https://github.com/novatorem/novatorem.git" | |||||
}, | |||||
"keywords": [], | |||||
"author": "Nate Moore", | |||||
"license": "ISC", | |||||
"bugs": { | |||||
"url": "https://github.com/novatorem/novatorem/issues" | |||||
}, | |||||
"homepage": "https://github.com/novatorem/novatorem#readme", | |||||
"husky": { | |||||
"hooks": { | |||||
"pre-commit": "lint-staged" | |||||
} | |||||
}, | |||||
"lint-staged": { | |||||
"*.{js,css,md}": "prettier --write" | |||||
} | |||||
} |
@ -0,0 +1,11 @@ | |||||
{ | |||||
"compilerOptions": { | |||||
"module": "ES2015", | |||||
"moduleResolution": "node", | |||||
"target": "ES2017", | |||||
"jsx": "react", | |||||
"esModuleInterop": true, | |||||
"skipLibCheck": true, | |||||
"forceConsistentCasingInFileNames": true | |||||
} | |||||
} |
@ -0,0 +1,46 @@ | |||||
import fetch from "isomorphic-unfetch"; | |||||
import { stringify } from "querystring"; | |||||
const { | |||||
SPOTIFY_CLIENT_ID: client_id, | |||||
SPOTIFY_CLIENT_SECRET: client_secret, | |||||
SPOTIFY_REFRESH_TOKEN: refresh_token, | |||||
} = process.env; | |||||
const basic = Buffer.from(`${client_id}:${client_secret}`).toString("base64"); | |||||
const Authorization = `Basic ${basic}`; | |||||
async function getAuthorizationToken() { | |||||
const url = new URL("https://accounts.spotify.com/api/token"); | |||||
const body = stringify({ | |||||
grant_type: "refresh_token", | |||||
refresh_token, | |||||
}); | |||||
const response = await fetch(`${url}`, { | |||||
method: "POST", | |||||
headers: { | |||||
Authorization, | |||||
"Content-Type": "application/x-www-form-urlencoded", | |||||
}, | |||||
body, | |||||
}).then((r) => r.json()); | |||||
return `Bearer ${response.access_token}`; | |||||
} | |||||
const NOW_PLAYING_ENDPOINT = `https://api.spotify.com/v1/me/player/currently-playing`; | |||||
export async function nowPlaying() { | |||||
const Authorization = await getAuthorizationToken(); | |||||
const response = await fetch(NOW_PLAYING_ENDPOINT, { | |||||
headers: { | |||||
Authorization, | |||||
}, | |||||
}); | |||||
const { status } = response; | |||||
if (status === 204) { | |||||
return {}; | |||||
} else if (status === 200) { | |||||
const data = await response.json(); | |||||
return data; | |||||
} | |||||
} |
@ -0,0 +1,4 @@ | |||||
{ | |||||
"version": 2, | |||||
"rewrites": [{ "source": "/(.*)", "destination": "/api/$1" }] | |||||
} |