<?php class spotify{ private const req_web = 0; private const req_api = 1; private const req_clientid = 2; public function __construct(){ include "lib/backend.php"; $this->backend = new backend("spotify"); include "lib/fuckhtml.php"; $this->fuckhtml = new fuckhtml(); } public function getfilters($page){ return [ "category" => [ "display" => "Category", "option" => [ "any" => "All (no pagination)", "audiobooks" => "Audiobooks", "tracks" => "Songs", "artists" => "Artists", "playlists" => "Playlists", "albums" => "Albums", "podcastAndEpisodes" => "Podcasts & Shows (no pagination)", "episodes" => "Episodes", "users" => "Profiles" ] ] ]; } private function get($proxy, $url, $get = [], $reqtype = self::req_web, $bearer = null, $token = null){ $curlproc = curl_init(); switch($reqtype){ case self::req_api: $headers = [ "User-Agent: " . config::USER_AGENT, "Accept: application/json", "Accept-Language: en", "app-platform: WebPlayer", "authorization: Bearer {$bearer}", "client-token: {$token}", "content-type: application/json;charset=UTF-8", "Origin: https://open.spotify.com", "Referer: https://open.spotify.com/", "DNT: 1", "Connection: keep-alive", "Sec-Fetch-Dest: empty", "Sec-Fetch-Mode: cors", "Sec-Fetch-Site: same-site", "spotify-app-version: 1.2.27.93.g7aee53d4", "TE: trailers" ]; break; case self::req_web: $headers = [ "User-Agent: " . config::USER_AGENT, "Accept: text/html,application/xhtml+xml,application/xml;q=0.9,image/avif,image/webp,*/*;q=0.8", "Accept-Language: en-US,en;q=0.5", "Accept-Encoding: gzip", "DNT: 1", "Sec-GPC: 1", "Connection: keep-alive", "Upgrade-Insecure-Requests: 1", "Sec-Fetch-Dest: document", "Sec-Fetch-Mode: navigate", "Sec-Fetch-Site: cross-site" ]; break; case self::req_clientid: $get = json_encode($get); curl_setopt($curlproc, CURLOPT_POST, true); curl_setopt($curlproc, CURLOPT_POSTFIELDS, $get); $headers = [ "User-Agent:" . config::USER_AGENT, "Accept: application/json", "Accept-Language: en-US,en;q=0.5", "Accept-Encoding: gzip, deflate, br", "Referer: https://open.spotify.com/", "content-type: application/json", "Content-Length: " . strlen($get), "Origin: https://open.spotify.com", "DNT: 1", "Sec-GPC: 1", "Connection: keep-alive", "Sec-Fetch-Dest: empty", "Sec-Fetch-Mode: cors", "Sec-Fetch-Site: same-site", "TE: trailers" ]; break; } if($reqtype !== self::req_clientid){ if($get !== []){ $get = http_build_query($get); $url .= "?" . $get; } } curl_setopt($curlproc, CURLOPT_URL, $url); curl_setopt($curlproc, CURLOPT_ENCODING, ""); // default encoding curl_setopt($curlproc, CURLOPT_HTTPHEADER, $headers); curl_setopt($curlproc, CURLOPT_RETURNTRANSFER, true); curl_setopt($curlproc, CURLOPT_SSL_VERIFYHOST, 2); curl_setopt($curlproc, CURLOPT_SSL_VERIFYPEER, true); curl_setopt($curlproc, CURLOPT_CONNECTTIMEOUT, 30); curl_setopt($curlproc, CURLOPT_TIMEOUT, 30); $this->backend->assign_proxy($curlproc, $proxy); $data = curl_exec($curlproc); if(curl_errno($curlproc)){ throw new Exception(curl_error($curlproc)); } curl_close($curlproc); return $data; } public function music($get){ $search = $get["s"]; $ip = $this->backend->get_ip(); $category = $get["category"]; /* audiobooks first and second page decoded https://api-partner.spotify.com/pathfinder/v1/query?operationName=searchAudiobooks&variables={"searchTerm":"freddie+dredd","offset":0,"limit":30,"numberOfTopResults":20,"includeAudiobooks":true}&extensions={"persistedQuery":{"version":1,"sha256Hash":"8758e540afdba5afa3c5246817f6bd31d86a15b3f5666c363dd017030f35d785"}} https://api-partner.spotify.com/pathfinder/v1/query?operationName=searchAudiobooks&variables={"searchTerm":"freddie+dredd","offset":30,"limit":30,"numberOfTopResults":20,"includeAudiobooks":true}&extensions={"persistedQuery":{"version":1,"sha256Hash":"8758e540afdba5afa3c5246817f6bd31d86a15b3f5666c363dd017030f35d785"}} */ /* songs https://api-partner.spotify.com/pathfinder/v1/query?operationName=searchTracks&variables={"searchTerm":"asmr","offset":0,"limit":100,"numberOfTopResults":20,"includeAudiobooks":false}&extensions={"persistedQuery":{"version":1,"sha256Hash":"16c02d6304f5f721fc2eb39dacf2361a4543815112506a9c05c9e0bc9733a679"}} https://api-partner.spotify.com/pathfinder/v1/query?operationName=searchTracks&variables={"searchTerm":"asmr","offset":100,"limit":100,"numberOfTopResults":20,"includeAudiobooks":false}&extensions={"persistedQuery":{"version":1,"sha256Hash":"16c02d6304f5f721fc2eb39dacf2361a4543815112506a9c05c9e0bc9733a679"}} */ /* artists https://api-partner.spotify.com/pathfinder/v1/query?operationName=searchArtists&variables={"searchTerm":"asmr","offset":0,"limit":30,"numberOfTopResults":20,"includeAudiobooks":true}&extensions={"persistedQuery":{"version":1,"sha256Hash":"b8840daafdda9a9ceadb7c5774731f63f9eca100445d2d94665f2dc58b45e2b9"}} https://api-partner.spotify.com/pathfinder/v1/query?operationName=searchArtists&variables={"searchTerm":"asmr","offset":30,"limit":23,"numberOfTopResults":20,"includeAudiobooks":true}&extensions={"persistedQuery":{"version":1,"sha256Hash":"b8840daafdda9a9ceadb7c5774731f63f9eca100445d2d94665f2dc58b45e2b9"}} https://api-partner.spotify.com/pathfinder/v1/query?operationName=searchArtists&variables={"searchTerm":"asmr","offset":53,"limit":30,"numberOfTopResults":20,"includeAudiobooks":true}&extensions={"persistedQuery":{"version":1,"sha256Hash":"b8840daafdda9a9ceadb7c5774731f63f9eca100445d2d94665f2dc58b45e2b9"}} */ /* playlists https://api-partner.spotify.com/pathfinder/v1/query?operationName=searchPlaylists&variables={"searchTerm":"asmr","offset":0,"limit":30,"numberOfTopResults":20,"includeAudiobooks":true}&extensions={"persistedQuery":{"version":1,"sha256Hash":"19b4143a0500ccec189ca0f4a0316bc2c615ecb51ce993ba4d7d08afd1d87aa4"}} https://api-partner.spotify.com/pathfinder/v1/query?operationName=searchPlaylists&variables={"searchTerm":"asmr","offset":30,"limit":3,"numberOfTopResults":20,"includeAudiobooks":true}&extensions={"persistedQuery":{"version":1,"sha256Hash":"19b4143a0500ccec189ca0f4a0316bc2c615ecb51ce993ba4d7d08afd1d87aa4"}} */ /* albums https://api-partner.spotify.com/pathfinder/v1/query?operationName=searchAlbums&variables={"searchTerm":"asmr","offset":33,"limit":30,"numberOfTopResults":20,"includeAudiobooks":true}&extensions={"persistedQuery":{"version":1,"sha256Hash":"e93b13cda461482da2940467eb2beed9368e9bb2fff37df3fb6633fc61271a27"}} https://api-partner.spotify.com/pathfinder/v1/query?operationName=searchAlbums&variables={"searchTerm":"asmr","offset":33,"limit":30,"numberOfTopResults":20,"includeAudiobooks":true}&extensions={"persistedQuery":{"version":1,"sha256Hash":"e93b13cda461482da2940467eb2beed9368e9bb2fff37df3fb6633fc61271a27"}} */ /* podcasts & shows (contains authors, no pagination) https://api-partner.spotify.com/pathfinder/v1/query?operationName=searchFullEpisodes&variables={"searchTerm":"asmr","offset":0,"limit":30}&extensions={"persistedQuery":{"version":1,"sha256Hash":"9f996251c9781fabce63f1a9980b5287ea33bc5e8c8953d0c4689b09936067a1"}} */ /* episodes https://api-partner.spotify.com/pathfinder/v1/query?operationName=searchDesktop&variables={"searchTerm":"asmr","offset":0,"limit":10,"numberOfTopResults":5,"includeAudiobooks":true}&extensions={"persistedQuery":{"version":1,"sha256Hash":"da03293d92a2cfc5e24597dcdc652c0ad135e1c64a78fddbf1478a7e096bea44"}} ??? https://api-partner.spotify.com/pathfinder/v1/query?operationName=searchFullEpisodes&variables={"searchTerm":"asmr","offset":60,"limit":30}&extensions={"persistedQuery":{"version":1,"sha256Hash":"9f996251c9781fabce63f1a9980b5287ea33bc5e8c8953d0c4689b09936067a1"}} */ /* profiles https://api-partner.spotify.com/pathfinder/v1/query?operationName=searchUsers&variables={"searchTerm":"asmr","offset":0,"limit":30,"numberOfTopResults":20,"includeAudiobooks":true}&extensions={"persistedQuery":{"version":1,"sha256Hash":"02026f48ab5001894e598904079b620ebc64f2d53b55ca20c3858abd3a46c5fb"}} https://api-partner.spotify.com/pathfinder/v1/query?operationName=searchUsers&variables={"searchTerm":"asmr","offset":30,"limit":30,"numberOfTopResults":20,"includeAudiobooks":true}&extensions={"persistedQuery":{"version":1,"sha256Hash":"02026f48ab5001894e598904079b620ebc64f2d53b55ca20c3858abd3a46c5fb"}} */ // get HTML try{ $html = $this->get( $ip, "https://open.spotify.com/search/" . rawurlencode($search) . ($category != "any" ? "/" . $category : ""), [] ); }catch(Exception $error){ throw new Exception("Failed to get initial search page"); } // grep bearer and client ID $this->fuckhtml->load($html); $script = $this->fuckhtml ->getElementById( "session", "script" ); if($script === null){ throw new Exception("Failed to grep bearer token"); } $script = json_decode( $script["innerHTML"], true ); $bearer = $script["accessToken"]; $client_id = $script["clientId"]; // hit client ID endpoint try{ $token = json_decode( $this->get( $ip, "https://clienttoken.spotify.com/v1/clienttoken", [ // !! that shit must be sent as json data "client_data" => [ "client_id" => $client_id, "client_version" => "1.2.27.93.g7aee53d4", "js_sdk_data" => [ "device_brand" => "unknown", "device_id" => "4c7ca20117ca12288ea8fc7118a9118c", "device_model" => "unknown", "device_name" => "computer", "os" => "windows", "os_version" => "NT 10.0" ] ] ], self::req_clientid ), true ); }catch(Exception $error){ throw new Exception("Failed to fetch token"); } if($token === null){ throw new Exception("Failed to decode token"); } $token = $token["granted_token"]["token"]; try{ switch($get["option"]){ case "any": $variables = [ "searchTerm" => $search, "offset" => 0, "limit" => 10, "numberOfTopResults" => 5, "includeAudiobooks" => true ]; break; case "audiobooks": break; } $payload = $this->get( $ip, "https://api-partner.spotify.com/pathfinder/v1/query", [ "operationName" => "searchDesktop", "variables" => json_encode( [ "searchTerm" => $search, "offset" => 0, "limit" => 10, "numberOfTopResults" => 5, "includeAudiobooks" => true ] ), "extensions" => json_encode( [ "persistedQuery" => [ "version" => 1, "sha256Hash" => "21969b655b795601fb2d2204a4243188e75fdc6d3520e7b9cd3f4db2aff9591e" // ? ] ] ) ], self::req_api, $bearer, $token ); }catch(Exception $error){ throw new Exception("Failed to fetch JSON results"); } if($payload == "Token expired"){ throw new Exception("Grepped spotify token has expired"); } $payload = json_decode($payload, true); if($payload === null){ throw new Exception("Failed to decode JSON results"); } //$payload = json_decode(file_get_contents("scraper/spotify.json"), true); $out = [ "status" => "ok", "npt" => null, "song" => [], "playlist" => [], "album" => [], "podcast" => [], "author" => [], "user" => [] ]; // get songs foreach($payload["data"]["searchV2"]["tracksV2"]["items"] as $result){ if(isset($result["item"])){ $result = $result["item"]; } if(isset($result["data"])){ $result = $result["data"]; } [$artist, $artist_link] = $this->get_artists($result["artists"]); $out["song"][] = [ "title" => $result["name"], "description" => null, "url" => "https://open.spotify.com/track/" . $result["id"], "views" => null, "author" => [ "name" => $artist, "url" => $artist_link, "avatar" => null ], "thumb" => $this->get_thumb($result["albumOfTrack"]["coverArt"]), "date" => null, "duration" => $result["duration"]["totalMilliseconds"] / 1000, "stream" => [ "endpoint" => "spotify", "url" => "track." . $result["id"] ] ]; } // get playlists foreach($payload["data"]["searchV2"]["playlists"]["items"] as $playlist){ if(isset($playlist["data"])){ $playlist = $playlist["data"]; } $avatar = $this->get_thumb($playlist["ownerV2"]["data"]["avatar"]); $out["playlist"][] = [ "title" => $playlist["name"], "description" => null, "author" => [ "name" => $playlist["ownerV2"]["data"]["name"], "url" => "https://open.spotify.com/user/" . explode( ":", $playlist["ownerV2"]["data"]["uri"], 3 )[2], "avatar" => $avatar["url"] ], "thumb" => $this->get_thumb($playlist["images"]["items"][0]), "date" => null, "duration" => null, "url" => "https://open.spotify.com/playlist/" . explode( ":", $playlist["uri"], 3 )[2] ]; } // get albums foreach($payload["data"]["searchV2"]["albums"]["items"] as $album){ if(isset($album["data"])){ $album = $album["data"]; } [$artist, $artist_link] = $this->get_artists($album["artists"]); $out["album"][] = [ "title" => $album["name"], "description" => null, "author" => [ "name" => $artist, "url" => $artist_link, "avatar" => null ], "thumb" => $this->get_thumb($album["coverArt"]), "date" => mktime(0, 0, 0, 0, 32, $album["date"]["year"]), "duration" => null, "url" => "https://open.spotify.com/album/" . explode( ":", $album["uri"], 3 )[2] ]; } // get podcasts foreach($payload["data"]["searchV2"]["podcasts"]["items"] as $podcast){ if(isset($podcast["data"])){ $podcast = $podcast["data"]; } $description = []; foreach($podcast["topics"]["items"] as $subject){ $description[] = $subject["title"]; } $description = implode(", ", $description); if($description == ""){ $description = null; } $out["podcast"][] = [ "title" => $podcast["name"], "description" => $description, "author" => [ "name" => $podcast["publisher"]["name"], "url" => null, "avatar" => null ], "thumb" => $this->get_thumb($podcast["coverArt"]), "date" => null, "duration" => null, "url" => "https://open.spotify.com/show/" . explode( ":", $podcast["uri"], 3 )[2], "stream" => [ "endpoint" => null, "url" => null ] ]; } // get audio books (put in podcasts) foreach($payload["data"]["searchV2"]["audiobooks"]["items"] as $podcast){ if(isset($podcast["data"])){ $podcast = $podcast["data"]; } $description = []; foreach($podcast["topics"]["items"] as $subject){ $description[] = $subject["title"]; } $description = implode(", ", $description); if($description == ""){ $description = null; } $authors = []; foreach($podcast["authors"] as $author){ $authors[] = $author["name"]; } $authors = implode(", ", $authors); if($authors == ""){ $authors = null; } $uri = explode( ":", $podcast["uri"], 3 )[2]; $out["podcast"][] = [ "title" => $podcast["name"], "description" => $description, "author" => [ "name" => $authors, "url" => null, "avatar" => null ], "thumb" => $this->get_thumb($podcast["coverArt"]), "date" => strtotime($podcast["publishDate"]["isoString"]), "duration" => null, "url" => "https://open.spotify.com/show/" . $uri, "stream" => [ "endpoint" => "spotify", "url" => "episode." . $uri ] ]; } // get episodes (and place them in podcasts) foreach($payload["data"]["searchV2"]["episodes"]["items"] as $podcast){ if(isset($podcast["data"])){ $podcast = $podcast["data"]; } $out["podcast"][] = [ "title" => $podcast["name"], "description" => $this->limitstrlen($podcast["description"]), "author" => [ "name" => isset( $podcast["podcastV2"]["data"]["publisher"]["name"] ) ? $podcast["podcastV2"]["data"]["publisher"]["name"] : null, "url" => null, "avatar" => null ], "thumb" => $this->get_thumb($podcast["coverArt"]), "date" => strtotime($podcast["releaseDate"]["isoString"]), "duration" => $podcast["duration"]["totalMilliseconds"] / 1000, "url" => "https://open.spotify.com/show/" . explode( ":", $podcast["uri"], 3 )[2], "stream" => [ "endpoint" => null, "url" => null ] ]; } // get authors foreach($payload["data"]["searchV2"]["artists"]["items"] as $user){ if(isset($user["data"])){ $user = $user["data"]; } $avatar = $this->get_thumb($user["visuals"]["avatarImage"]); $out["author"][] = [ "title" => ( $user["profile"]["verified"] === true ? "✓ " : "" ) . $user["profile"]["name"], "followers" => null, "description" => null, "thumb" => $avatar, "url" => "https://open.spotify.com/artist/" . explode( ":", $user["uri"], 3 )[2] ]; } // get users foreach($payload["data"]["searchV2"]["users"]["items"] as $user){ if(isset($user["data"])){ $user = $user["data"]; } $avatar = $this->get_thumb($user["avatar"]); $out["user"][] = [ "title" => $user["displayName"] . " (@{$user["id"]})", "followers" => null, "description" => null, "thumb" => $avatar, "url" => "https://open.spotify.com/user/" . $user["id"] ]; } return $out; } private function get_artists($artists){ $artist_out = []; foreach($artists["items"] as $artist){ $artist_out[] = $artist["profile"]["name"]; } $artist_out = implode(", ", $artist_out); if($artist_out == ""){ return [null, null]; } $artist_link = $artist === null ? null : "https://open.spotify.com/artist/" . explode( ":", $artists["items"][0]["uri"] )[2]; return [$artist_out, $artist_link]; } private function get_thumb($cover){ $thumb_out = null; if($cover !== null){ foreach($cover["sources"] as $thumb){ if( $thumb_out === null || (int)$thumb["width"] > $thumb_out["width"] ){ $thumb_out = $thumb; } } } if($thumb_out === null){ return [ "url" => null, "ratio" => null ]; }else{ return [ "url" => $thumb_out["url"], "ratio" => "1:1" ]; } } private function limitstrlen($text){ return explode( "\n", wordwrap( str_replace( ["\n\r", "\r\n", "\n", "\r"], " ", $text ), 300, "\n" ), 2 )[0]; } }