<?php

if(!isset($_GET["s"])){
	
	header("X-Error: Missing parameter (s)ite");
	die();
}

include "data/config.php";
new favicon($_GET["s"]);

class favicon{
	
	public function __construct($url){
		
		header("Content-Type: image/png");
		
		if(substr_count($url, "/") !== 2){
			
			header("X-Error: Only provide the protocol and domain");
			$this->defaulticon();
		}
		
		$filename = str_replace(["https://", "http://"], "", $url);
		header("Content-Disposition: inline; filename=\"{$filename}.png\"");
		
		include "lib/curlproxy.php";
		$this->proxy = new proxy(false);
		
		$this->filename = parse_url($url, PHP_URL_HOST);
		
		/*
			Check if we have the favicon stored locally
		*/
		if(file_exists("icons/" . $filename . ".png")){
				
			$handle = fopen("icons/" . $filename . ".png", "r");
			echo fread($handle, filesize("icons/" . $filename . ".png"));
			fclose($handle);
			return;
		}
		
		/*
			Scrape html
		*/
		try{
			
			$payload = $this->proxy->get($url, $this->proxy::req_web, true);
			
		}catch(Exception $error){
			
			header("X-Error: Could not fetch HTML (" . $error->getMessage() . ")");
			$this->favicon404();
		}
		//$payload["body"] = '<link rel="manifest" id="MANIFEST_LINK" href="/data/manifest/" crossorigin="use-credentials" />';
		
		// get link tags
		preg_match_all(
			'/< *link +(.*)[\/]?>/Uixs',
			$payload["body"],
			$linktags
		);
		
		/*
			Get relevant tags
		*/
		
		$linktags = $linktags[1];
		$attributes = [];
		
		/*
		header("Content-Type: text/plain");
		print_r($linktags);
		print_r($payload);
		die();*/
		
		for($i=0; $i<count($linktags); $i++){
			
			// get attributes
			preg_match_all(
				'/([A-Za-z0-9]+) *= *("[^"]*"|[^" ]+)/s',
				$linktags[$i],
				$tags
			);
			
			for($k=0; $k<count($tags[1]); $k++){
				
				$attributes[$i][] = [
					"name" => $tags[1][$k],
					"value" => trim($tags[2][$k], "\" \n\r\t\v\x00")
				];
			}
		}

		unset($payload);
		unset($linktags);

		$href = [];
		
		// filter out the tags we want
		foreach($attributes as &$group){
			
			$tmp_href = null;
			$tmp_rel = null;
			$badtype = false;
			
			foreach($group as &$attribute){
				
				switch($attribute["name"]){
					
					case "rel":
						
						$attribute["value"] = strtolower($attribute["value"]);
						
						if(
							(
								$attribute["value"] == "icon" ||
								$attribute["value"] == "manifest" ||
								$attribute["value"] == "shortcut icon" ||
								$attribute["value"] == "apple-touch-icon" ||
								$attribute["value"] == "mask-icon"
							) === false
						){
							
							break;
						}
						
						$tmp_rel = $attribute["value"];
						break;
					
					case "type":
						$attribute["value"] = explode("/", $attribute["value"], 2);
						
						if(strtolower($attribute["value"][0]) != "image"){
							
							$badtype = true;
							break;
						}
						break;
					
					case "href":
						
						// must not contain invalid characters
						// must be bigger than 1
						if(
							filter_var($attribute["value"], FILTER_SANITIZE_URL) == $attribute["value"] &&
							strlen($attribute["value"]) > 0
						){
							
							$tmp_href = $attribute["value"];
							break;
						}
						break;
				}
			}
			
			if(
				$badtype === false &&
				$tmp_rel !== null &&
				$tmp_href !== null
			){
				
				$href[$tmp_rel] = $tmp_href;
			}
		}
		
		/*
			Priority list
		*/
		/*
		header("Content-Type: text/plain");
		print_r($href);
		die();*/
		
		if(isset($href["icon"])){ $href = $href["icon"]; }
		elseif(isset($href["apple-touch-icon"])){ $href = $href["apple-touch-icon"]; }
		elseif(isset($href["manifest"])){
			
			// attempt to parse manifest, but fallback to []
			$href = $this->parsemanifest($href["manifest"], $url);
		}
		
		if(is_array($href)){
			
			if(isset($href["mask-icon"])){ $href = $href["mask-icon"]; }
			elseif(isset($href["shortcut icon"])){ $href = $href["shortcut icon"]; }
			else{
				
				$href = "/favicon.ico";
			}
		}
		
		$href = $this->proxy->getabsoluteurl($href, $url);
		/*
		header("Content-type: text/plain");
		echo $href;
		die();*/
		
		
		/*
			Download the favicon
		*/
		//$href = "https://git.lolcat.ca/assets/img/logo.svg";
		
		try{
			$payload =
				$this->proxy->get(
					$href,
					$this->proxy::req_image,
					true,
					$url
				);
				
		}catch(Exception $error){
			
			header("X-Error: Could not fetch the favicon (" . $error->getMessage() . ")");
			$this->favicon404();
		}
		
		/*
			Parse the file format
		*/
		$image = null;
		$format = $this->proxy->getimageformat($payload, $image);
		
		/*
			Convert the image
		*/
		try{
			
			/*
				@todo: fix issues with avif+transparency
				maybe using GD as fallback?
			*/
			if($format !== false){
				$image->setFormat($format);
			}
			
			$image->setBackgroundColor(new ImagickPixel("transparent"));
			$image->readImageBlob($payload["body"]);
			$image->resizeImage(16, 16, imagick::FILTER_LANCZOS, 1);
			$image->setFormat("png");
			
			$image = $image->getImageBlob();
			
			// save favicon
			$handle = fopen("icons/" . $this->filename . ".png", "w");
			fwrite($handle, $image, strlen($image));
			fclose($handle);
			
			echo $image;
			
		}catch(ImagickException $error){
			
			header("X-Error: Could not convert the favicon: (" . $error->getMessage() . ")");
			$this->favicon404();
		}
		
		return;
	}
	
	private function parsemanifest($href, $url){
		
		if(
			// check if base64-encoded JSON manifest
			preg_match(
				'/^data:application\/json;base64,([A-Za-z0-9=]*)$/',
				$href,
				$json
			)
		){
			
			$json = base64_decode($json[1]);
			
			if($json === false){
				
				// could not decode the manifest regex
				return [];
			}
			
		}else{
			
			try{
				$json =
					$this->proxy->get(
						$this->proxy->getabsoluteurl($href, $url),
						$this->proxy::req_web,
						false,
						$url
					);
					
					$json = $json["body"];
					
			}catch(Exception $error){
				
				// could not fetch the manifest
				return [];
			}
		}
		
		$json = json_decode($json, true);
		
		if($json === null){
			
			// manifest did not return valid json
			return [];
		}
		
		if(
			isset($json["start_url"]) &&
			$this->proxy->validateurl($json["start_url"])
		){
			
			$url = $json["start_url"];
		}
		
		if(!isset($json["icons"][0]["src"])){
			
			// manifest does not contain a path to the favicon
			return [];
		}
		
		// horay, return the favicon path
		return $json["icons"][0]["src"];
	}
	
	private function favicon404(){
		
		// fallback to google favicons
		// ... probably blocked by cuckflare
		try{
			
			$image =
				$this->proxy->get(
					"https://t0.gstatic.com/faviconV2?client=SOCIAL&type=FAVICON&fallback_opts=TYPE,SIZE,URL&url=http://{$this->filename}&size=16",
					$this->proxy::req_image
				);
		}catch(Exception $error){
			
			$this->defaulticon();
		}
		
		// write favicon from google
		$handle = fopen("icons/" . $this->filename . ".png", "w");
		fwrite($handle, $image["body"], strlen($image["body"]));
		fclose($handle);
		
		echo $image["body"];
		die();
	}
	
	private function defaulticon(){
		
		// give 404 and fuck off
		http_response_code(404);
		
		$handle = fopen("lib/favicon404.png", "r");
		echo fread($handle, filesize("lib/favicon404.png"));
		fclose($handle);
		
		die();
	}
}