mirror of
				https://github.com/jakejarvis/jarv.is.git
				synced 2025-10-26 01:05:49 -04:00 
			
		
		
		
	restore the /api/playing endpoint
This commit is contained in:
		
							
								
								
									
										141
									
								
								pages/api/playing.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										141
									
								
								pages/api/playing.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,141 @@ | ||||
| // Fetches my Spotify most-played tracks or currently playing track. | ||||
| // Heavily inspired by @leerob: https://leerob.io/snippets/spotify | ||||
|  | ||||
| import queryString from "query-string"; | ||||
| import { logServerError } from "../../lib/helpers/sentry"; | ||||
| import type { NextApiRequest, NextApiResponse } from "next"; | ||||
|  | ||||
| const { SPOTIFY_CLIENT_ID, SPOTIFY_CLIENT_SECRET, SPOTIFY_REFRESH_TOKEN } = process.env; | ||||
|  | ||||
| // https://developer.spotify.com/documentation/general/guides/authorization-guide/#authorization-code-flow | ||||
| const TOKEN_ENDPOINT = "https://accounts.spotify.com/api/token"; | ||||
| // https://developer.spotify.com/documentation/web-api/reference/#endpoint-get-the-users-currently-playing-track | ||||
| const NOW_PLAYING_ENDPOINT = "https://api.spotify.com/v1/me/player/currently-playing"; | ||||
| // https://developer.spotify.com/documentation/web-api/reference/#endpoint-get-users-top-artists-and-tracks | ||||
| const TOP_TRACKS_ENDPOINT = "https://api.spotify.com/v1/me/top/tracks?time_range=long_term&limit=10"; | ||||
|  | ||||
| type Track = { | ||||
|   artist: string; | ||||
|   title: string; | ||||
|   album: string; | ||||
|   url: URL | string; | ||||
|   image?: URL | string; | ||||
| }; | ||||
|  | ||||
| type SpotifyTrackSchema = { | ||||
|   name: string; | ||||
|   artists: Array<{ | ||||
|     name: string; | ||||
|   }>; | ||||
|   album: { | ||||
|     name: string; | ||||
|     images?: Array<{ | ||||
|       url: URL | string; | ||||
|     }>; | ||||
|   }; | ||||
|   imageUrl?: URL | string; | ||||
|   external_urls: { | ||||
|     spotify: URL | string; | ||||
|   }; | ||||
| }; | ||||
|  | ||||
| const handler = async (req: NextApiRequest, res: NextApiResponse) => { | ||||
|   try { | ||||
|     if (req.method !== "GET") { | ||||
|       // 405 Method Not Allowed | ||||
|       return res.status(405).end(); | ||||
|     } | ||||
|  | ||||
|     // let Vercel edge cache results for 5 mins | ||||
|     res.setHeader("Cache-Control", "public, max-age=0, s-maxage=300, stale-while-revalidate"); | ||||
|  | ||||
|     const token = await getAccessToken(); | ||||
|     const playing = await getNowPlaying(token); | ||||
|     const top = await getTopTracks(token); | ||||
|  | ||||
|     return res.status(200).json({ playing, top }); | ||||
|   } catch (error) { | ||||
|     const message = error instanceof Error ? error.message : "Unknown error."; | ||||
|  | ||||
|     // log full error to console and sentry | ||||
|     await logServerError(error); | ||||
|  | ||||
|     // 500 Internal Server Error | ||||
|     return res.status(500).json({ success: false, message }); | ||||
|   } | ||||
| }; | ||||
|  | ||||
| const getAccessToken = async () => { | ||||
|   const basic = Buffer.from(`${SPOTIFY_CLIENT_ID}:${SPOTIFY_CLIENT_SECRET}`).toString("base64"); | ||||
|  | ||||
|   const response = await fetch(TOKEN_ENDPOINT, { | ||||
|     method: "POST", | ||||
|     headers: { | ||||
|       Authorization: `Basic ${basic}`, | ||||
|       "Content-Type": "application/x-www-form-urlencoded", | ||||
|     }, | ||||
|     body: queryString.stringify({ | ||||
|       grant_type: "refresh_token", | ||||
|       refresh_token: SPOTIFY_REFRESH_TOKEN, | ||||
|     }), | ||||
|   }); | ||||
|  | ||||
|   const { access_token: token } = await response.json(); | ||||
|  | ||||
|   return token as string; | ||||
| }; | ||||
|  | ||||
| const getNowPlaying = async (token: string): Promise<Track | false> => { | ||||
|   const response = await fetch(NOW_PLAYING_ENDPOINT, { | ||||
|     headers: { | ||||
|       Authorization: `Bearer ${token}`, | ||||
|       Accept: "application/json", | ||||
|       "Content-Type": "application/json", | ||||
|     }, | ||||
|   }); | ||||
|  | ||||
|   if (response.status === 204 || response.status > 400) { | ||||
|     return false; | ||||
|   } | ||||
|  | ||||
|   const active = (await response.json()) as { | ||||
|     is_playing: boolean; | ||||
|     item?: SpotifyTrackSchema; | ||||
|   }; | ||||
|  | ||||
|   if (active?.is_playing === true && active?.item) { | ||||
|     return { | ||||
|       artist: active.item.artists.map((artist) => artist.name).join(", "), | ||||
|       title: active.item.name, | ||||
|       album: active.item.album.name, | ||||
|       image: active.item.album.images ? active.item.album.images[0].url : undefined, | ||||
|       url: active.item.external_urls.spotify, | ||||
|     }; | ||||
|   } | ||||
|  | ||||
|   return false; | ||||
| }; | ||||
|  | ||||
| const getTopTracks = async (token: string): Promise<Track[]> => { | ||||
|   const response = await fetch(TOP_TRACKS_ENDPOINT, { | ||||
|     headers: { | ||||
|       Authorization: `Bearer ${token}`, | ||||
|       Accept: "application/json", | ||||
|       "Content-Type": "application/json", | ||||
|     }, | ||||
|   }); | ||||
|  | ||||
|   const { items } = (await response.json()) as { items: SpotifyTrackSchema[] }; | ||||
|  | ||||
|   const tracks: Track[] = items.map((track: SpotifyTrackSchema) => ({ | ||||
|     artist: track.artists.map((artist) => artist.name).join(", "), | ||||
|     title: track.name, | ||||
|     album: track.album.name, | ||||
|     image: track.album.images ? track.album.images[0].url : undefined, | ||||
|     url: track.external_urls.spotify, | ||||
|   })); | ||||
|  | ||||
|   return tracks; | ||||
| }; | ||||
|  | ||||
| export default handler; | ||||
		Reference in New Issue
	
	Block a user