<?php class frontend{ public function load($template, $replacements = []){ $replacements["server_name"] = htmlspecialchars(config::SERVER_NAME); $replacements["version"] = config::VERSION; if(isset($_COOKIE["theme"])){ $theme = str_replace(["/". "."], "", $_COOKIE["theme"]); if( $theme != "Dark" && !is_file("static/themes/" . $theme . ".css") ){ $theme = config::DEFAULT_THEME; } }else{ $theme = config::DEFAULT_THEME; } if($theme != "Dark"){ $replacements["style"] = '<link rel="stylesheet" href="/static/themes/' . rawurlencode($theme) . '.css?v' . config::VERSION . '">'; }else{ $replacements["style"] = ""; } if(isset($_COOKIE["scraper_ac"])){ $replacements["ac"] = '?ac=' . htmlspecialchars($_COOKIE["scraper_ac"]); }else{ $replacements["ac"] = ''; } if( isset($replacements["timetaken"]) && $replacements["timetaken"] !== null ){ $replacements["timetaken"] = '<div class="timetaken">Took ' . number_format(microtime(true) - $replacements["timetaken"], 2) . 's</div>'; } $handle = fopen("template/{$template}", "r"); $data = fread($handle, filesize("template/{$template}")); fclose($handle); $data = explode("\n", $data); $html = ""; for($i=0; $i<count($data); $i++){ $html .= trim($data[$i]); } foreach($replacements as $key => $value){ $html = str_replace( "{%{$key}%}", $value, $html ); } return trim($html); } public function loadheader(array $get, array $filters, string $page){ echo $this->load("header.html", [ "title" => trim(htmlspecialchars($get["s"]) . " ({$page})"), "description" => ucfirst($page) . ' search results for "' . htmlspecialchars($get["s"]) . '"', "index" => "no", "search" => htmlspecialchars($get["s"]), "tabs" => $this->generatehtmltabs($page, $get["s"]), "filters" => $this->generatehtmlfilters($filters, $get) ]); $headers_raw = getallheaders(); $header_keys = []; $user_agent = ""; $bad_header = false; foreach($headers_raw as $headerkey => $headervalue){ $headerkey = strtolower($headerkey); if($headerkey == "user-agent"){ $user_agent = $headervalue; continue; } // check header key if(in_array($headerkey, config::FILTERED_HEADER_KEYS)){ $bad_header = true; break; } } if( preg_match( config::HEADER_REGEX, $user_agent ) || $bad_header === true ){ // bot detected !! apcu_inc("captcha_gen"); $this->drawerror( "Tshh, blocked!", 'Your browser, IP or IP range has been blocked from this 4get instance. If this is an error, please <a href="/about">contact the administrator</a>.' ); die(); } } public function drawerror($title, $error, $timetaken = null){ if($timetaken === null){ $timetaken = microtime(true); } echo $this->load("search.html", [ "timetaken" => $timetaken, "class" => "", "right-left" => "", "right-right" => "", "left" => '<div class="infobox">' . '<h1>' . htmlspecialchars($title) . '</h1>' . $error . '</div>' ]); die(); } public function drawscrapererror($error, $get, $target, $timetaken = null){ if($timetaken === null){ $timetaken = microtime(true); } $this->drawerror( "Shit", 'This scraper returned an error:' . '<div class="code">' . htmlspecialchars($error) . '</div>' . 'Things you can try:' . '<ul>' . '<li>Use a different scraper</li>' . '<li>Remove keywords that could cause errors</li>' . '<li><a href="/instances?target=' . $target . "&" . $this->buildquery($get, false) . '">Try your search on another 4get instance</a></li>' . '</ul><br>' . 'If the error persists, please <a href="/about">contact the administrator</a>.', $timetaken ); } public function drawtextresult($site, $greentext = null, $duration = null, $keywords, $tabindex = true, $customhtml = null){ $payload = '<div class="text-result">'; // add favicon, link and archive links $payload .= $this->drawlink($site["url"]); /* Draw title + description + filetype */ $payload .= '<a href="' . htmlspecialchars($site["url"]) . '" class="hover" rel="noreferrer nofollow"'; if($tabindex === false){ $payload .= ' tabindex="-1"'; } $payload .= '>'; if($site["thumb"]["url"] !== null){ $payload .= '<div class="thumb-wrap'; switch($site["thumb"]["ratio"]){ case "16:9": $size = "landscape"; break; case "9:16": $payload .= " portrait"; $size = "portrait"; break; case "1:1": $payload .= " square"; $size = "square"; break; } $payload .= '">' . '<img class="thumb" src="' . $this->htmlimage($site["thumb"]["url"], $size) . '" alt="thumb">'; if($duration !== null){ $payload .= '<div class="duration">' . htmlspecialchars($duration) . '</div>'; } $payload .= '</div>'; } $payload .= '<div class="title">'; if( isset($site["type"]) && $site["type"] != "web" ){ $payload .= '<div class="type">' . strtoupper($site["type"]) . '</div>'; } $payload .= $this->highlighttext($keywords, $site["title"]) . '</div>'; if($greentext !== null){ $payload .= '<div class="greentext">' . htmlspecialchars($greentext) . '</div>'; } if($site["description"] !== null){ $payload .= '<div class="description">' . $this->highlighttext($keywords, $site["description"]) . '</div>'; } $payload .= $customhtml; $payload .= '</a>'; /* Sublinks */ if( isset($site["sublink"]) && !empty($site["sublink"]) ){ usort($site["sublink"], function($a, $b){ return strlen($a["description"]) > strlen($b["description"]); }); $payload .= '<div class="sublinks">' . '<table>'; $opentr = false; for($i=0; $i<count($site["sublink"]); $i++){ if(($i % 2) === 0){ $opentr = true; $payload .= '<tr>'; }else{ $opentr = false; } $payload .= '<td>' . '<a href="' . htmlspecialchars($site["sublink"][$i]["url"]) . '" rel="noreferrer nofollow">' . '<div class="title">' . htmlspecialchars($site["sublink"][$i]["title"]) . '</div>'; if(!empty($site["sublink"][$i]["date"])){ $payload .= '<div class="greentext">' . date("jS M y @ g:ia", $site["sublink"][$i]["date"]) . '</div>'; } if(!empty($site["sublink"][$i]["description"])){ $payload .= '<div class="description">' . $this->highlighttext($keywords, $site["sublink"][$i]["description"]) . '</div>'; } $payload .= '</a></td>'; if($opentr === false){ $payload .= '</tr>'; } } if($opentr === true){ $payload .= '<td></td></tr>'; } $payload .= '</table></div>'; } if( isset($site["table"]) && !empty($site["table"]) ){ $payload .= '<table class="info-table">'; foreach($site["table"] as $title => $value){ $payload .= '<tr>' . '<td>' . htmlspecialchars($title) . '</td>' . '<td>' . htmlspecialchars($value) . '</td>' . '</tr>'; } $payload .= '</table>'; } return $payload . '</div>'; } public function highlighttext($keywords, $text){ $text = htmlspecialchars($text); $keywords = explode(" ", $keywords); $regex = []; foreach($keywords as $word){ $regex[] = "\b" . preg_quote($word, "/") . "\b"; } $regex = "/" . implode("|", $regex) . "/i"; return preg_replace( $regex, '<b>${0}</b>', $text ); } function highlightcode($text){ // https://www.php.net/highlight_string ini_set("highlight.comment", "c-comment"); ini_set("highlight.default", "c-default"); ini_set("highlight.html", "c-default"); ini_set("highlight.keyword", "c-keyword"); ini_set("highlight.string", "c-string"); $text = trim( preg_replace( '/<\/span>$/', "", // remove stray ending span because of the <?php stuff str_replace( [ '<br />', ' ' ], [ "\n", // replace <br> with newlines " " // replace html entity to space ], str_replace( [ // leading <?php garbage "<span style=\"color: c-default\">\n<?php ", "<code>", "</code>" ], "", highlight_string("<?php " . $text, true) ) ) ) ); // replace colors $classes = ["c-comment", "c-default", "c-keyword", "c-string"]; foreach($classes as $class){ $text = str_replace('<span style="color: ' . $class . '">', '<span class="' . $class . '">', $text); } return $text; } public function drawlink($link){ /* Add favicon */ $host = parse_url($link); $esc = explode( ".", $host["host"], 2 ); if( count($esc) === 2 && $esc[0] == "www" ){ $esc = $esc[1]; }else{ $esc = $esc[0]; } $esc = substr($esc, 0, 2); $urlencode = urlencode($link); $payload = '<div class="url">' . '<button class="favicon" tabindex="-1">' . '<img src="/favicon?s=' . htmlspecialchars($host["scheme"] . "://" . $host["host"]) . '" alt="' . htmlspecialchars($esc) . '">' . //'<img src="/404.php" alt="' . htmlspecialchars($esc) . '">' . '</button>' . '<div class="favicon-dropdown">'; /* Add archive links */ if( $host["host"] == "boards.4chan.org" || $host["host"] == "boards.4channel.org" ){ $archives = []; $path = explode("/", $host["path"]); $count = count($path); // /pol/thread/417568063/post-shitty-memes-if-you-want-to if($count !== 0){ $isboard = true; switch($path[1]){ case "con": break; case "q": $archives[] = "desuarchive.org"; break; case "qa": $archives[] = "desuarchive.org"; break; case "qb": $archives[] = "arch.b4k.co"; break; case "trash": $archives[] = "desuarchive.org"; break; case "a": $archives[] = "desuarchive.org"; break; case "c": $archives[] = "desuarchive.org"; break; case "w": break; case "m": $archives[] = "desuarchive.org"; break; case "cgl": $archives[] = "desuarchive.org"; $archives[] = "warosu.org"; break; case "f": $archives[] = "archive.4plebs.org"; break; case "n": break; case "jp": $archives[] = "warosu.org"; break; case "vt": $archives[] = "warosu.org"; break; case "v": $archives[] = "arch.b4k.co"; break; case "vg": $archives[] = "arch.b4k.co"; break; case "vm": $archives[] = "arch.b4k.co"; break; case "vmg": $archives[] = "arch.b4k.co"; break; case "vp": $archives[] = "arch.b4k.co"; break; case "vr": $archives[] = "desuarchive.org"; $archives[] = "warosu.org"; break; case "vrpg": $archives[] = "arch.b4k.co"; break; case "vst": $archives[] = "arch.b4k.co"; break; case "co": $archives[] = "desuarchive.org"; break; case "g": $archives[] = "desuarchive.org"; $archives[] = "arch.b4k.co"; break; case "tv": $archives[] = "archive.4plebs.org"; break; case "k": $archives[] = "desuarchive.org"; break; case "o": $archives[] = "archive.4plebs.org"; break; case "an": $archives[] = "desuarchive.org"; break; case "tg": $archives[] = "desuarchive.org"; $archives[] = "archive.4plebs.org"; break; case "sp": $archives[] = "archive.4plebs.org"; break; case "xs": $archives[] = "eientei.xyz"; break; case "pw": break; case "sci": $archives[] = "warosu.org"; $archives[] = "eientei.xyz"; break; case "his": $archives[] = "desuarchive.org"; break; case "int": $archives[] = "desuarchive.org"; break; case "out": break; case "toy": break; case "i": $archives[] = "archiveofsins.com"; $archives[] = "eientei.xyz"; break; case "po": break; case "p": break; case "ck": $archives[] = "warosu.org"; break; case "ic": $archives[] = "warosu.org"; break; case "wg": break; case "lit": $archives[] = "warosu.org"; break; case "mu": $archives[] = "desuarchive.org"; break; case "fa": $archives[] = "warosu.org"; break; case "3": $archives[] = "warosu.org"; $archives[] = "eientei.xyz"; break; case "gd": break; case "diy": $archives[] = "warosu.org"; break; case "wsg": $archives[] = "desuarchive.org"; break; case "qst": break; case "biz": $archives[] = "warosu.org"; break; case "trv": $archives[] = "archive.4plebs.org"; break; case "fit": $archives[] = "desuarchive.org"; break; case "x": $archives[] = "archive.4plebs.org"; break; case "adv": $archives[] = "archive.4plebs.org"; break; case "lgbt": $archives[] = "archiveofsins.com"; break; case "mlp": $archives[] = "desuarchive.org"; $archives[] = "arch.b4k.co"; break; case "news": break; case "wsr": break; case "vip": break; case "b": $archives[] = "thebarchive.com"; break; case "r9k": $archives[] = "desuarchive.org"; break; case "pol": $archives[] = "archive.4plebs.org"; break; case "bant": $archives[] = "thebarchive.com"; break; case "soc": $archives[] = "archiveofsins.com"; break; case "s4s": $archives[] = "archive.4plebs.org"; break; case "s": $archives[] = "archiveofsins.com"; break; case "hc": $archives[] = "archiveofsins.com"; break; case "hm": $archives[] = "archiveofsins.com"; break; case "h": $archives[] = "archiveofsins.com"; break; case "e": break; case "u": $archives[] = "archiveofsins.com"; break; case "d": $archives[] = "desuarchive.org"; break; case "t": $archives[] = "archiveofsins.com"; break; case "hr": $archives[] = "archive.4plebs.org"; break; case "gif": break; case "aco": $archives[] = "desuarchive.org"; break; case "r": $archives[] = "archiveofsins.com"; break; default: $isboard = false; break; } if($isboard === true){ $archives[] = "archived.moe"; } $trail = ""; if( isset($path[2]) && isset($path[3]) && $path[2] == "thread" ){ $trail .= "/" . $path[1] . "/thread/" . $path[3]; }elseif($isboard){ $trail = "/" . $path[1] . "/"; } for($i=0; $i<count($archives); $i++){ $payload .= '<a href="https://' . $archives[$i] . $trail . '" class="list" target="_BLANK">' . '<img src="/favicon?s=https://' . $archives[$i] . '" alt="' . $archives[$i][0] . $archives[$i][1] . '">' . $archives[$i] . '</a>'; } } } $payload .= '<a href="https://webcache.googleusercontent.com/search?q=cache:' . $urlencode . '" class="list" target="_BLANK"><img src="/favicon?s=https://google.com" alt="go">Google cache</a>' . '<a href="https://web.archive.org/web/' . $urlencode . '" class="list" target="_BLANK"><img src="/favicon?s=https://archive.org" alt="ar">Archive.org</a>' . '<a href="https://archive.ph/newest/' . htmlspecialchars($link) . '" class="list" target="_BLANK"><img src="/favicon?s=https://archive.is" alt="ar">Archive.is</a>' . '<a href="https://ghostarchive.org/search?term=' . $urlencode . '" class="list" target="_BLANK"><img src="/favicon?s=https://ghostarchive.org" alt="gh">Ghostarchive</a>' . '<a href="https://www.bing.com/search?q=url%3A' . $urlencode . '" class="list" target="_BLANK"><img src="/favicon?s=https://bing.com" alt="bi">Bing cache</a>' . '<a href="https://megalodon.jp/?url=' . $urlencode . '" class="list" target="_BLANK"><img src="/favicon?s=https://megalodon.jp" alt="me">Megalodon</a>' . '</div>'; /* Draw link */ $parts = explode("/", $link); $clickurl = ""; // remove trailing / $c = count($parts) - 1; if($parts[$c] == ""){ $parts[$c - 1] = $parts[$c - 1] . "/"; unset($parts[$c]); } // merge https://site together $parts = [ $parts[0] . $parts[1] . '//' . $parts[2], ...array_slice($parts, 3, count($parts) - 1) ]; $c = count($parts); for($i=0; $i<$c; $i++){ if($i !== 0){ $clickurl .= "/"; } $clickurl .= $parts[$i]; if($i === $c - 1){ $parts[$i] = rtrim($parts[$i], "/"); } $payload .= '<a class="part" href="' . htmlspecialchars($clickurl) . '" rel="noreferrer nofollow" tabindex="-1">' . htmlspecialchars(urldecode($parts[$i])) . '</a>'; if($i !== $c - 1){ $payload .= '<span class="separator"></span>'; } } return $payload . '</div>'; } public function getscraperfilters($page){ $get_scraper = isset($_COOKIE["scraper_$page"]) ? $_COOKIE["scraper_$page"] : null; if( isset($_GET["scraper"]) && is_string($_GET["scraper"]) ){ $get_scraper = $_GET["scraper"]; }else{ if( isset($_GET["npt"]) && is_string($_GET["npt"]) ){ $get_scraper = explode(".", $_GET["npt"], 2)[0]; $get_scraper = preg_replace( '/[0-9]+$/', "", $get_scraper ); } } // add search field $filters = [ "s" => [ "option" => "_SEARCH" ] ]; // define default scrapers switch($page){ case "web": $filters["scraper"] = [ "display" => "Scraper", "option" => [ "ddg" => "DuckDuckGo", "brave" => "Brave", "yandex" => "Yandex", "google" => "Google", "qwant" => "Qwant", "yep" => "Yep", "greppr" => "Greppr", "crowdview" => "Crowdview", "mwmbl" => "Mwmbl", "mojeek" => "Mojeek", "marginalia" => "Marginalia", "wiby" => "wiby", "curlie" => "Curlie" ] ]; break; case "images": $filters["scraper"] = [ "display" => "Scraper", "option" => [ "ddg" => "DuckDuckGo", "yandex" => "Yandex", "brave" => "Brave", "google" => "Google", "qwant" => "Qwant", "yep" => "Yep", //"pinterest" => "Pinterest", "imgur" => "Imgur", "ftm" => "FindThatMeme" ] ]; break; case "videos": $filters["scraper"] = [ "display" => "Scraper", "option" => [ "yt" => "YouTube", //"fb" => "Facebook videos", "ddg" => "DuckDuckGo", "brave" => "Brave", "yandex" => "Yandex", "google" => "Google", "qwant" => "Qwant" ] ]; break; case "news": $filters["scraper"] = [ "display" => "Scraper", "option" => [ "ddg" => "DuckDuckGo", "brave" => "Brave", "google" => "Google", "qwant" => "Qwant", "yep" => "Yep", "mojeek" => "Mojeek" ] ]; break; case "music": $filters["scraper"] = [ "display" => "Scraper", "option" => [ "sc" => "SoundCloud" //"spotify" => "Spotify" ] ]; break; } // get scraper name from user input, or default out to preferred scraper $scraper_out = null; $first = true; foreach($filters["scraper"]["option"] as $scraper_name => $scraper_pretty){ if($first === true){ $first = $scraper_name; } if($scraper_name == $get_scraper){ $scraper_out = $scraper_name; } } if($scraper_out === null){ $scraper_out = $first; } include "scraper/$scraper_out.php"; $lib = new $scraper_out(); // set scraper on $_GET $_GET["scraper"] = $scraper_out; // set nsfw on $_GET if( isset($_COOKIE["nsfw"]) && !isset($_GET["nsfw"]) ){ $_GET["nsfw"] = $_COOKIE["nsfw"]; } return [ $lib, array_merge_recursive( $filters, $lib->getfilters($page) ) ]; } public function parsegetfilters($parameters, $whitelist){ $sanitized = []; // add npt token if( isset($parameters["npt"]) && is_string($parameters["npt"]) ){ $sanitized["npt"] = $parameters["npt"]; }else{ $sanitized["npt"] = false; } // we're iterating over $whitelist, so // you can't polluate $sanitized with useless // parameters foreach($whitelist as $parameter => $value){ if(isset($parameters[$parameter])){ if(!is_string($parameters[$parameter])){ $sanitized[$parameter] = null; continue; } // parameter is already set, use that value $sanitized[$parameter] = $parameters[$parameter]; }else{ // parameter is not set, add it if(is_string($value["option"])){ // special field: set default value manually switch($value["option"]){ case "_DATE": // no date set $sanitized[$parameter] = false; break; case "_SEARCH": // no search set $sanitized[$parameter] = ""; break; } }else{ // set a default value $sanitized[$parameter] = array_keys($value["option"])[0]; } } // sanitize input if(is_array($value["option"])){ if( !in_array( $sanitized[$parameter], $keys = array_keys($value["option"]) ) ){ $sanitized[$parameter] = $keys[0]; } }else{ // sanitize search & string switch($value["option"]){ case "_DATE": if($sanitized[$parameter] !== false){ $sanitized[$parameter] = strtotime($sanitized[$parameter]); if($sanitized[$parameter] <= 0){ $sanitized[$parameter] = false; } } break; case "_SEARCH": // get search string $sanitized["s"] = trim($sanitized[$parameter]); } } } // invert dates if needed if( isset($sanitized["older"]) && isset($sanitized["newer"]) && $sanitized["newer"] !== false && $sanitized["older"] !== false && $sanitized["newer"] > $sanitized["older"] ){ // invert [ $sanitized["older"], $sanitized["newer"] ] = [ $sanitized["newer"], $sanitized["older"] ]; } return $sanitized; } public function s_to_timestamp($seconds){ if(is_string($seconds)){ return "LIVE"; } return ($seconds >= 60) ? ltrim(gmdate("H:i:s", $seconds), ":0") : gmdate("0:s", $seconds); } public function generatehtmltabs($page, $query){ $html = null; foreach(["web", "images", "videos", "news", "music"] as $type){ $html .= '<a href="/' . $type . '?s=' . urlencode($query); if(!empty($params)){ $html .= $params; } $html .= '" class="tab'; if($type == $page){ $html .= ' selected'; } $html .= '">' . ucfirst($type) . '</a>'; } return $html; } public function generatehtmlfilters($filters, $params){ $html = null; foreach($filters as $filter_name => $filter_values){ if(!isset($filter_values["display"])){ continue; } $output = true; $tmp = '<div class="filter">' . '<div class="title">' . htmlspecialchars($filter_values["display"]) . '</div>'; if(is_array($filter_values["option"])){ $tmp .= '<select name="' . $filter_name . '">'; foreach($filter_values["option"] as $option_name => $option_title){ $tmp .= '<option value="' . $option_name . '"'; if($params[$filter_name] == $option_name){ $tmp .= ' selected'; } $tmp .= '>' . htmlspecialchars($option_title) . '</option>'; } $tmp .= '</select>'; }else{ switch($filter_values["option"]){ case "_DATE": $tmp .= '<input type="date" name="' . $filter_name . '"'; if($params[$filter_name] !== false){ $tmp .= ' value="' . date("Y-m-d", $params[$filter_name]) . '"'; } $tmp .= '>'; break; default: $output = false; break; } } $tmp .= '</div>'; if($output === true){ $html .= $tmp; } } return $html; } public function buildquery($gets, $ommit = false){ $out = []; foreach($gets as $key => $value){ if( $value == null || $value == false || $key == "npt" || $key == "extendedsearch" || $value == "any" || $value == "all" || $key == "spellcheck" || ( $ommit === true && $key == "s" ) ){ continue; } if( $key == "older" || $key == "newer" ){ $value = date("Y-m-d", (int)$value); } $out[$key] = $value; } return http_build_query($out); } public function htmlimage($image, $format){ if( preg_match( '/^data:/', $image ) ){ return htmlspecialchars($image); } return "/proxy?i=" . urlencode($image) . "&s=" . $format; } public function htmlnextpage($gets, $npt, $page){ $query = $this->buildquery($gets); return $page . "?" . $query . "&npt=" . $npt; } }