Browse Source

Switch to TypeScript/React

add-license-1
novatorem 5 years ago
parent
commit
afcda3f742
14 changed files with 1597 additions and 231 deletions
  1. +4
    -0
      .gitignore
  2. +3
    -1
      README.md
  3. +0
    -11
      api/date.py
  4. +44
    -0
      api/now-playing.ts
  5. +0
    -3
      api/requirements.txt
  6. +0
    -216
      api/spotify-playing.py
  7. +162
    -0
      components/NowPlaying.tsx
  8. +27
    -0
      components/ReadmeImg.tsx
  9. +51
    -0
      components/Text.tsx
  10. +1202
    -0
      package-lock.json
  11. +43
    -0
      package.json
  12. +11
    -0
      tsconfig.json
  13. +46
    -0
      utils/spotify.ts
  14. +4
    -0
      vercel.json

+ 4
- 0
.gitignore View File

@ -0,0 +1,4 @@
.vercel
.env
node_modules
.DS_STORE

+ 3
- 1
README.md View File

@ -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>

+ 0
- 11
api/date.py View File

@ -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

+ 44
- 0
api/now-playing.ts View File

@ -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);
}

+ 0
- 3
api/requirements.txt View File

@ -1,3 +0,0 @@
flask==1.1.2
requests==2.24.0
python-dotenv==0.14.0

+ 0
- 216
api/spotify-playing.py View File

@ -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("&", "&amp;")
song_name = item["name"].replace("&", "&amp;")
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)

+ 162
- 0
components/NowPlaying.tsx View File

@ -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>
);
};

+ 27
- 0
components/ReadmeImg.tsx View File

@ -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;

+ 51
- 0
components/Text.tsx View File

@ -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;

+ 1202
- 0
package-lock.json
File diff suppressed because it is too large
View File


+ 43
- 0
package.json View File

@ -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"
}
}

+ 11
- 0
tsconfig.json View File

@ -0,0 +1,11 @@
{
"compilerOptions": {
"module": "ES2015",
"moduleResolution": "node",
"target": "ES2017",
"jsx": "react",
"esModuleInterop": true,
"skipLibCheck": true,
"forceConsistentCasingInFileNames": true
}
}

+ 46
- 0
utils/spotify.ts View File

@ -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;
}
}

+ 4
- 0
vercel.json View File

@ -0,0 +1,4 @@
{
"version": 2,
"rewrites": [{ "source": "/(.*)", "destination": "/api/$1" }]
}

Loading…
Cancel
Save