1175 lines
24 KiB
PHP
1175 lines
24 KiB
PHP
<?php
|
|
|
|
class mojeek{
|
|
public function __construct(){
|
|
|
|
include "lib/fuckhtml.php";
|
|
$this->fuckhtml = new fuckhtml();
|
|
|
|
include "lib/backend.php";
|
|
$this->backend = new backend("mojeek");
|
|
}
|
|
|
|
public function getfilters($page){
|
|
|
|
switch($page){
|
|
|
|
case "web":
|
|
return [
|
|
"focus" => [
|
|
"display" => "Focus",
|
|
"option" => [
|
|
"any" => "No focus",
|
|
"blogs" => "Blogs",
|
|
"Dictionary" => "Dictionary",
|
|
"Recipes" => "Recipes",
|
|
"Time" => "Time",
|
|
"Weather" => "Weather"
|
|
]
|
|
],
|
|
"lang" => [
|
|
"display" => "Language",
|
|
"option" => [
|
|
"any" => "Any language",
|
|
"af" => "Afrikaans",
|
|
"sq" => "Albanian",
|
|
"an" => "Aragonese",
|
|
"ay" => "Aymara",
|
|
"bi" => "Bislama",
|
|
"br" => "Breton",
|
|
"ca" => "Catalan",
|
|
"kw" => "Cornish",
|
|
"co" => "Corsican",
|
|
"hr" => "Croatian",
|
|
"da" => "Danish",
|
|
"nl" => "Dutch",
|
|
"dz" => "Dzongkha",
|
|
"en" => "English",
|
|
"fj" => "Fijian",
|
|
"fi" => "Finnish",
|
|
"fr" => "French",
|
|
"gd" => "Gaelic",
|
|
"gl" => "Galician",
|
|
"de" => "German",
|
|
"ht" => "Haitian",
|
|
"io" => "Ido",
|
|
"id" => "Indonesian",
|
|
"ia" => "Interlingua",
|
|
"ie" => "Interlingue",
|
|
"ga" => "Irish",
|
|
"it" => "Italian",
|
|
"rw" => "Kinyarwanda",
|
|
"la" => "Latin",
|
|
"li" => "Limburgish",
|
|
"lb" => "Luxembourgish",
|
|
"no" => "Norwegian",
|
|
"nb" => "Norwegian Bokmål",
|
|
"nn" => "Norwegian Nynorsk",
|
|
"oc" => "Occitan (post 1500)",
|
|
"pl" => "Polish",
|
|
"pt" => "Portuguese",
|
|
"rm" => "Romansh",
|
|
"rn" => "Rundi",
|
|
"sg" => "Sango",
|
|
"so" => "Somali",
|
|
"es" => "Spanish",
|
|
"sw" => "Swahili",
|
|
"ss" => "Swati",
|
|
"sv" => "Swedish",
|
|
"ty" => "Tahitian",
|
|
"to" => "Tonga (Tonga Islands)",
|
|
"ts" => "Tsonga",
|
|
"vo" => "Volapük",
|
|
"wa" => "Walloon",
|
|
"cy" => "Welsh",
|
|
"xh" => "Xhosa",
|
|
"zu" => "Zulu"
|
|
]
|
|
],
|
|
"country" => [
|
|
"display" => "Country",
|
|
"option" => [
|
|
"any" => "No location bias",
|
|
"af" => "Afghanistan",
|
|
"ax" => "Åland Islands",
|
|
"al" => "Albania",
|
|
"dz" => "Algeria",
|
|
"as" => "American Samoa",
|
|
"ad" => "Andorra",
|
|
"ao" => "Angola",
|
|
"ai" => "Anguilla",
|
|
"aq" => "Antarctica",
|
|
"ag" => "Antigua and Barbuda",
|
|
"ar" => "Argentina",
|
|
"am" => "Armenia",
|
|
"aw" => "Aruba",
|
|
"au" => "Australia",
|
|
"at" => "Austria",
|
|
"az" => "Azerbaijan",
|
|
"bs" => "Bahamas",
|
|
"bh" => "Bahrain",
|
|
"bd" => "Bangladesh",
|
|
"bb" => "Barbados",
|
|
"by" => "Belarus",
|
|
"be" => "Belgium",
|
|
"bz" => "Belize",
|
|
"bj" => "Benin",
|
|
"bm" => "Bermuda",
|
|
"bt" => "Bhutan",
|
|
"bo" => "Bolivia (Plurinational State of)",
|
|
"bq" => "Bonaire, Sint Eustatius and Saba",
|
|
"ba" => "Bosnia and Herzegovina",
|
|
"bw" => "Botswana",
|
|
"bv" => "Bouvet Island",
|
|
"br" => "Brazil",
|
|
"io" => "British Indian Ocean Territory",
|
|
"bn" => "Brunei Darussalam",
|
|
"bg" => "Bulgaria",
|
|
"bf" => "Burkina Faso",
|
|
"bi" => "Burundi",
|
|
"cv" => "Cabo Verde",
|
|
"kh" => "Cambodia",
|
|
"cm" => "Cameroon",
|
|
"ca" => "Canada",
|
|
"ky" => "Cayman Islands",
|
|
"cf" => "Central African Republic",
|
|
"td" => "Chad",
|
|
"cl" => "Chile",
|
|
"cn" => "China",
|
|
"cx" => "Christmas Island",
|
|
"cc" => "Cocos (Keeling) Islands",
|
|
"co" => "Colombia",
|
|
"km" => "Comoros",
|
|
"cg" => "Congo",
|
|
"cd" => "Congo (Democratic Republic of the)",
|
|
"ck" => "Cook Islands",
|
|
"cr" => "Costa Rica",
|
|
"ci" => "Côte d'Ivoire",
|
|
"hr" => "Croatia",
|
|
"cu" => "Cuba",
|
|
"cw" => "Curaçao",
|
|
"cy" => "Cyprus",
|
|
"cz" => "Czechia",
|
|
"dk" => "Denmark",
|
|
"dj" => "Djibouti",
|
|
"dm" => "Dominica",
|
|
"do" => "Dominican Republic",
|
|
"ec" => "Ecuador",
|
|
"eg" => "Egypt",
|
|
"sv" => "El Salvador",
|
|
"gq" => "Equatorial Guinea",
|
|
"er" => "Eritrea",
|
|
"ee" => "Estonia",
|
|
"et" => "Ethiopia",
|
|
"fk" => "Falkland Islands (Malvinas)",
|
|
"fo" => "Faroe Islands",
|
|
"fj" => "Fiji",
|
|
"fi" => "Finland",
|
|
"fr" => "France",
|
|
"gf" => "French Guiana",
|
|
"pf" => "French Polynesia",
|
|
"tf" => "French Southern Territories",
|
|
"ga" => "Gabon",
|
|
"gm" => "Gambia",
|
|
"ge" => "Georgia",
|
|
"de" => "Germany",
|
|
"gh" => "Ghana",
|
|
"gi" => "Gibraltar",
|
|
"gr" => "Greece",
|
|
"gl" => "Greenland",
|
|
"gd" => "Grenada",
|
|
"gp" => "Guadeloupe",
|
|
"gu" => "Guam",
|
|
"gt" => "Guatemala",
|
|
"gg" => "Guernsey",
|
|
"gn" => "Guinea",
|
|
"gw" => "Guinea-Bissau",
|
|
"gy" => "Guyana",
|
|
"ht" => "Haiti",
|
|
"hm" => "Heard Island and McDonald Islands",
|
|
"va" => "Holy See",
|
|
"hn" => "Honduras",
|
|
"hk" => "Hong Kong",
|
|
"hu" => "Hungary",
|
|
"is" => "Iceland",
|
|
"in" => "India",
|
|
"id" => "Indonesia",
|
|
"ir" => "Iran (Islamic Republic of)",
|
|
"iq" => "Iraq",
|
|
"ie" => "Ireland",
|
|
"im" => "Isle of Man",
|
|
"il" => "Israel",
|
|
"it" => "Italy",
|
|
"jm" => "Jamaica",
|
|
"jp" => "Japan",
|
|
"je" => "Jersey",
|
|
"jo" => "Jordan",
|
|
"kz" => "Kazakhstan",
|
|
"ke" => "Kenya",
|
|
"ki" => "Kiribati",
|
|
"kp" => "Korea (Democratic People's Republic of)",
|
|
"kr" => "Korea (Republic of)",
|
|
"kw" => "Kuwait",
|
|
"kg" => "Kyrgyzstan",
|
|
"la" => "Lao People's Democratic Republic",
|
|
"lv" => "Latvia",
|
|
"lb" => "Lebanon",
|
|
"ls" => "Lesotho",
|
|
"lr" => "Liberia",
|
|
"ly" => "Libya",
|
|
"li" => "Liechtenstein",
|
|
"lt" => "Lithuania",
|
|
"lu" => "Luxembourg",
|
|
"mo" => "Macao",
|
|
"mk" => "Macedonia (the former Yugoslav Republic of)",
|
|
"mg" => "Madagascar",
|
|
"mw" => "Malawi",
|
|
"my" => "Malaysia",
|
|
"mv" => "Maldives",
|
|
"ml" => "Mali",
|
|
"mt" => "Malta",
|
|
"mh" => "Marshall Islands",
|
|
"mq" => "Martinique",
|
|
"mr" => "Mauritania",
|
|
"mu" => "Mauritius",
|
|
"yt" => "Mayotte",
|
|
"mx" => "Mexico",
|
|
"fm" => "Micronesia (Federated States of)",
|
|
"md" => "Moldova (Republic of)",
|
|
"mc" => "Monaco",
|
|
"mn" => "Mongolia",
|
|
"me" => "Montenegro",
|
|
"ms" => "Montserrat",
|
|
"ma" => "Morocco",
|
|
"mz" => "Mozambique",
|
|
"mm" => "Myanmar",
|
|
"na" => "Namibia",
|
|
"nr" => "Nauru",
|
|
"np" => "Nepal",
|
|
"nl" => "Netherlands",
|
|
"nc" => "New Caledonia",
|
|
"nz" => "New Zealand",
|
|
"ni" => "Nicaragua",
|
|
"ne" => "Niger",
|
|
"ng" => "Nigeria",
|
|
"nu" => "Niue",
|
|
"nf" => "Norfolk Island",
|
|
"mp" => "Northern Mariana Islands",
|
|
"no" => "Norway",
|
|
"om" => "Oman",
|
|
"pk" => "Pakistan",
|
|
"pw" => "Palau",
|
|
"ps" => "Palestine, State of",
|
|
"pa" => "Panama",
|
|
"pg" => "Papua New Guinea",
|
|
"py" => "Paraguay",
|
|
"pe" => "Peru",
|
|
"ph" => "Philippines",
|
|
"pn" => "Pitcairn",
|
|
"pl" => "Poland",
|
|
"pt" => "Portugal",
|
|
"pr" => "Puerto Rico",
|
|
"qa" => "Qatar",
|
|
"re" => "Réunion",
|
|
"ro" => "Romania",
|
|
"ru" => "Russian Federation",
|
|
"rw" => "Rwanda",
|
|
"bl" => "Saint Barthélemy",
|
|
"sh" => "Saint Helena, Ascension and Tristan da Cunha",
|
|
"kn" => "Saint Kitts and Nevis",
|
|
"lc" => "Saint Lucia",
|
|
"mf" => "Saint Martin (French part)",
|
|
"pm" => "Saint Pierre and Miquelon",
|
|
"vc" => "Saint Vincent and the Grenadines",
|
|
"ws" => "Samoa",
|
|
"sm" => "San Marino",
|
|
"st" => "Sao Tome and Principe",
|
|
"sa" => "Saudi Arabia",
|
|
"sn" => "Senegal",
|
|
"rs" => "Serbia",
|
|
"sc" => "Seychelles",
|
|
"sl" => "Sierra Leone",
|
|
"sg" => "Singapore",
|
|
"sx" => "Sint Maarten (Dutch part)",
|
|
"sk" => "Slovakia",
|
|
"si" => "Slovenia",
|
|
"sb" => "Solomon Islands",
|
|
"so" => "Somalia",
|
|
"za" => "South Africa",
|
|
"gs" => "South Georgia and South Sandwich Islands",
|
|
"ss" => "South Sudan",
|
|
"es" => "Spain",
|
|
"lk" => "Sri Lanka",
|
|
"sd" => "Sudan",
|
|
"sr" => "Suriname",
|
|
"sj" => "Svalbard and Jan Mayen",
|
|
"sz" => "Swaziland",
|
|
"se" => "Sweden",
|
|
"ch" => "Switzerland",
|
|
"sy" => "Syrian Arab Republic",
|
|
"tw" => "Taiwan",
|
|
"tj" => "Tajikistan",
|
|
"tz" => "Tanzania, United Republic of",
|
|
"th" => "Thailand",
|
|
"tl" => "Timor-Leste",
|
|
"tg" => "Togo",
|
|
"tk" => "Tokelau",
|
|
"to" => "Tonga",
|
|
"tt" => "Trinidad and Tobago",
|
|
"tn" => "Tunisia",
|
|
"tr" => "Turkey",
|
|
"tm" => "Turkmenistan",
|
|
"tc" => "Turks and Caicos Islands",
|
|
"tv" => "Tuvalu",
|
|
"ug" => "Uganda",
|
|
"ua" => "Ukraine",
|
|
"ae" => "United Arab Emirates",
|
|
"gb" => "United Kingdom",
|
|
"us" => "United States of America",
|
|
"um" => "United States Minor Outlying Islands",
|
|
"uy" => "Uruguay",
|
|
"uz" => "Uzbekistan",
|
|
"vu" => "Vanuatu",
|
|
"ve" => "Venezuela (Bolivarian Republic of)",
|
|
"vn" => "Viet Nam",
|
|
"vg" => "Virgin Islands (British)",
|
|
"vi" => "Virgin Islands (U.S.)",
|
|
"wf" => "Wallis and Futuna",
|
|
"eh" => "Western Sahara",
|
|
"ye" => "Yemen",
|
|
"zm" => "Zambia",
|
|
"zw" => "Zimbabwe"
|
|
]
|
|
],
|
|
"region" => [
|
|
"display" => "Region",
|
|
"option" => [
|
|
"any" => "Any region",
|
|
"eu" => "European Union",
|
|
"de" => "Germany",
|
|
"fr" => "France",
|
|
"uk" => "United Kingdom"
|
|
]
|
|
],
|
|
"domain" => [
|
|
"display" => "Results per domain",
|
|
"option" => [
|
|
"1" => "1 result",
|
|
"2" => "2 results",
|
|
"3" => "3 results",
|
|
"4" => "4 results",
|
|
"5" => "5 results",
|
|
"10" => "10 results",
|
|
"0" => "Unlimited",
|
|
]
|
|
]
|
|
];
|
|
break;
|
|
|
|
case "news":
|
|
return [];
|
|
}
|
|
}
|
|
|
|
private function get($proxy, $url, $get = []){
|
|
|
|
$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",
|
|
"Connection: keep-alive",
|
|
"Upgrade-Insecure-Requests: 1",
|
|
"Sec-Fetch-Dest: document",
|
|
"Sec-Fetch-Mode: navigate",
|
|
"Sec-Fetch-Site: none",
|
|
"Sec-Fetch-User: ?1"
|
|
];
|
|
|
|
$curlproc = curl_init();
|
|
|
|
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 web($get){
|
|
|
|
if($get["npt"]){
|
|
|
|
[$token, $proxy] = $this->backend->get($get["npt"], "web");
|
|
|
|
try{
|
|
$html =
|
|
$this->get(
|
|
$proxy,
|
|
"https://www.mojeek.com" . $token,
|
|
[]
|
|
);
|
|
}catch(Exception $error){
|
|
|
|
throw new Exception("Failed to get HTML");
|
|
}
|
|
|
|
}else{
|
|
|
|
$search = $get["s"];
|
|
if(strlen($search) === 0){
|
|
|
|
throw new Exception("Search term is empty!");
|
|
}
|
|
|
|
$proxy = $this->backend->get_ip();
|
|
$lang = $get["lang"];
|
|
$country = $get["country"];
|
|
$region = $get["region"];
|
|
$domain = $get["domain"];
|
|
$focus = $get["focus"];
|
|
|
|
$params = [
|
|
"q" => $search,
|
|
"t" => 20, // number of results/page
|
|
"tn" => 7, // number of news results/page
|
|
"date" => 1, // show date
|
|
"tlen" => 128, // max length of title
|
|
"dlen" => 511, // max length of description
|
|
"arc" => ($country == "any" ? "none" : $country) // location. don't use autodetect!
|
|
];
|
|
|
|
switch($focus){
|
|
|
|
case "any": break;
|
|
|
|
case "blogs":
|
|
$params["fmt"] = "sst";
|
|
$params["sst"] = "1";
|
|
break;
|
|
|
|
default:
|
|
$params["foc_t"] = $focus;
|
|
break;
|
|
}
|
|
|
|
if($lang != "any"){
|
|
|
|
$params["lb"] = $lang;
|
|
}
|
|
|
|
if($region != "any"){
|
|
|
|
$params["reg"] = $region;
|
|
}
|
|
|
|
if($domain != "1"){
|
|
|
|
$params["si"] = $domain;
|
|
}
|
|
|
|
try{
|
|
$html =
|
|
$this->get(
|
|
$proxy,
|
|
"https://www.mojeek.com/search",
|
|
$params
|
|
);
|
|
}catch(Exception $error){
|
|
|
|
throw new Exception("Failed to get HTML");
|
|
}
|
|
/*
|
|
$handle = fopen("scraper/mojeek.html", "r");
|
|
$html = fread($handle, filesize("scraper/mojeek.html"));
|
|
fclose($handle);*/
|
|
|
|
}
|
|
|
|
$out = [
|
|
"status" => "ok",
|
|
"spelling" => [
|
|
"type" => "no_correction",
|
|
"using" => null,
|
|
"correction" => null
|
|
],
|
|
"npt" => null,
|
|
"answer" => [],
|
|
"web" => [],
|
|
"image" => [],
|
|
"video" => [],
|
|
"news" => [],
|
|
"related" => []
|
|
];
|
|
|
|
$this->fuckhtml->load($html);
|
|
|
|
$results =
|
|
$this->fuckhtml
|
|
->getElementsByClassName("results-standard", "ul");
|
|
|
|
if(count($results) === 0){
|
|
|
|
return $out;
|
|
}
|
|
|
|
/*
|
|
Get all search result divs
|
|
*/
|
|
foreach($results as $container){
|
|
|
|
$this->fuckhtml->load($container);
|
|
$results =
|
|
$this->fuckhtml
|
|
->getElementsByTagName("li");
|
|
|
|
foreach($results as $result){
|
|
|
|
$data = [
|
|
"title" => null,
|
|
"description" => null,
|
|
"url" => null,
|
|
"date" => null,
|
|
"type" => "web",
|
|
"thumb" => [
|
|
"url" => null,
|
|
"ratio" => null
|
|
],
|
|
"sublink" => [],
|
|
"table" => []
|
|
];
|
|
|
|
$this->fuckhtml->load($result);
|
|
|
|
$title =
|
|
$this->fuckhtml
|
|
->getElementsByClassName("title", "a")[0];
|
|
|
|
$data["title"] =
|
|
html_entity_decode(
|
|
$this->fuckhtml
|
|
->getTextContent(
|
|
$title["innerHTML"]
|
|
)
|
|
);
|
|
|
|
$data["url"] =
|
|
html_entity_decode(
|
|
$this->fuckhtml
|
|
->getTextContent(
|
|
$title["attributes"]["href"]
|
|
)
|
|
);
|
|
|
|
$description =
|
|
$this->fuckhtml
|
|
->getElementsByClassName(
|
|
"s", "p"
|
|
);
|
|
|
|
if(count($description) !== 0){
|
|
|
|
$data["description"] =
|
|
$this->titledots(
|
|
html_entity_decode(
|
|
$this->fuckhtml
|
|
->getTextContent(
|
|
$description[0]
|
|
)
|
|
)
|
|
);
|
|
}
|
|
|
|
$date =
|
|
$this->fuckhtml
|
|
->getElementsByClassName(
|
|
"mdate",
|
|
"span"
|
|
);
|
|
|
|
if(count($date) !== 0){
|
|
|
|
$data["date"] =
|
|
strtotime(
|
|
$this->fuckhtml
|
|
->getTextContent(
|
|
$date[0]
|
|
)
|
|
);
|
|
}
|
|
|
|
$out["web"][] = $data;
|
|
}
|
|
}
|
|
|
|
/*
|
|
Get instant answers
|
|
*/
|
|
$this->fuckhtml->load($html);
|
|
|
|
$infoboxes =
|
|
$this->fuckhtml
|
|
->getElementsByClassName(
|
|
"infobox infobox-top",
|
|
"div"
|
|
);
|
|
|
|
foreach($infoboxes as $infobox){
|
|
|
|
$answer = [
|
|
"title" => null,
|
|
"description" => [],
|
|
"url" => null,
|
|
"thumb" => null,
|
|
"table" => [],
|
|
"sublink" => []
|
|
];
|
|
|
|
// load first part with title + short definition
|
|
$infobox_html =
|
|
explode(
|
|
"<hr>",
|
|
$infobox["innerHTML"]
|
|
);
|
|
|
|
$this->fuckhtml->load($infobox_html[0]);
|
|
|
|
// title
|
|
$answer["title"] =
|
|
$this->fuckhtml
|
|
->getTextContent(
|
|
$this->fuckhtml
|
|
->getElementsByTagName("h1")[0]
|
|
);
|
|
|
|
// short definition
|
|
$definition =
|
|
$this->fuckhtml
|
|
->getElementsByTagName(
|
|
"p"
|
|
);
|
|
|
|
if(count($definition) !== 0){
|
|
|
|
$answer["description"][] = [
|
|
"type" => "quote",
|
|
"value" =>
|
|
$this->fuckhtml
|
|
->getTextContent(
|
|
$definition[0]
|
|
)
|
|
];
|
|
}
|
|
|
|
// get thumbnail, if it exists
|
|
$this->fuckhtml->load($infobox_html[1]);
|
|
|
|
$thumb =
|
|
$this->fuckhtml
|
|
->getElementsByClassName("float-right", "img");
|
|
|
|
if(count($thumb) !== 0){
|
|
|
|
preg_match(
|
|
'/\/image\?img=([^&]+)/i',
|
|
$thumb[0]["attributes"]["src"],
|
|
$thumb
|
|
);
|
|
|
|
if(count($thumb) === 2){
|
|
|
|
$answer["thumb"] =
|
|
urldecode(
|
|
$this->fuckhtml
|
|
->getTextContent(
|
|
$thumb[1]
|
|
)
|
|
);
|
|
}
|
|
}
|
|
|
|
// get description
|
|
$ps =
|
|
$this->fuckhtml
|
|
->getElementsByTagName("p");
|
|
|
|
$first_tag = true;
|
|
foreach($ps as $p){
|
|
|
|
$this->fuckhtml->load($p);
|
|
|
|
if(
|
|
preg_match(
|
|
'/^\s*<strong>/i',
|
|
$p["innerHTML"]
|
|
)
|
|
){
|
|
|
|
/*
|
|
Parse table
|
|
*/
|
|
|
|
$strong =
|
|
$this->fuckhtml
|
|
->getElementsByTagName("strong")[0];
|
|
|
|
$p["innerHTML"] =
|
|
str_replace($strong["innerHTML"], "", $p["innerHTML"]);
|
|
|
|
$strong =
|
|
preg_replace(
|
|
'/:$/',
|
|
"",
|
|
ucfirst(
|
|
$this->fuckhtml
|
|
->getTextContent(
|
|
$strong
|
|
)
|
|
)
|
|
);
|
|
|
|
$answer["table"][trim($strong)] =
|
|
trim(
|
|
$this->fuckhtml
|
|
->getTextContent(
|
|
$p
|
|
)
|
|
);
|
|
|
|
continue;
|
|
}
|
|
|
|
$as =
|
|
$this->fuckhtml
|
|
->getElementsByClassName("svg-icon");
|
|
|
|
if(count($as) !== 0){
|
|
|
|
/*
|
|
Parse websites
|
|
*/
|
|
foreach($as as $a){
|
|
|
|
$answer["sublink"][
|
|
ucfirst(explode(" ", $a["attributes"]["class"], 2)[1])
|
|
] =
|
|
$this->fuckhtml
|
|
->getTextContent(
|
|
$a["attributes"]["href"]
|
|
);
|
|
}
|
|
|
|
continue;
|
|
}
|
|
|
|
/*
|
|
Parse text content
|
|
*/
|
|
$tags =
|
|
$this->fuckhtml
|
|
->getElementsByTagName("*");
|
|
|
|
$i = 0;
|
|
foreach($tags as $tag){
|
|
|
|
$c = count($answer["description"]);
|
|
|
|
// remove tag from innerHTML
|
|
$p["innerHTML"] =
|
|
explode($tag["outerHTML"], $p["innerHTML"], 2);
|
|
|
|
if(count($p["innerHTML"]) === 2){
|
|
|
|
if(
|
|
$i === 0 &&
|
|
$c !== 0 &&
|
|
$answer["description"][$c - 1]["type"] == "link"
|
|
){
|
|
|
|
$append = "\n\n";
|
|
}else{
|
|
|
|
$append = "";
|
|
}
|
|
|
|
if($p["innerHTML"][0] != ""){
|
|
$answer["description"][] = [
|
|
"type" => "text",
|
|
"value" => $append . trim($p["innerHTML"][0])
|
|
];
|
|
}
|
|
|
|
$p["innerHTML"] = $p["innerHTML"][1];
|
|
}else{
|
|
|
|
$p["innerHTML"] = $p["innerHTML"][0];
|
|
}
|
|
|
|
switch($tag["tagName"]){
|
|
|
|
case "a":
|
|
|
|
$value =
|
|
$this->fuckhtml
|
|
->getTextContent(
|
|
$tag
|
|
);
|
|
|
|
if(strtolower($value) == "wikipedia"){
|
|
|
|
if($c !== 0){
|
|
$answer["description"][$c - 1]["value"] =
|
|
rtrim($answer["description"][$c - 1]["value"]);
|
|
}
|
|
break;
|
|
}
|
|
|
|
$answer["description"][] = [
|
|
"type" => "link",
|
|
"url" =>
|
|
$this->fuckhtml
|
|
->getTextContent(
|
|
$tag["attributes"]["href"]
|
|
),
|
|
"value" =>
|
|
$this->fuckhtml
|
|
->getTextContent(
|
|
$tag
|
|
)
|
|
];
|
|
break;
|
|
}
|
|
|
|
$i++;
|
|
}
|
|
}
|
|
|
|
// get URL
|
|
$this->fuckhtml->load($infobox_html[2]);
|
|
|
|
$answer["url"] =
|
|
$this->fuckhtml
|
|
->getTextContent(
|
|
$this->fuckhtml
|
|
->getElementsByTagName(
|
|
"a"
|
|
)[0]
|
|
["attributes"]
|
|
["href"]
|
|
);
|
|
|
|
// append answer
|
|
$out["answer"][] = $answer;
|
|
}
|
|
|
|
/*
|
|
Get news
|
|
*/
|
|
$this->fuckhtml->load($html);
|
|
|
|
$news =
|
|
$this->fuckhtml
|
|
->getElementsByClassName(
|
|
"results news-results",
|
|
"div"
|
|
);
|
|
|
|
if(count($news) !== 0){
|
|
|
|
$this->fuckhtml->load($news[0]);
|
|
|
|
$lis =
|
|
$this->fuckhtml
|
|
->getElementsByTagName("li");
|
|
|
|
foreach($lis as $li){
|
|
|
|
$this->fuckhtml->load($li);
|
|
|
|
$a =
|
|
$this->fuckhtml
|
|
->getElementsByClassName(
|
|
"ob",
|
|
"a"
|
|
);
|
|
|
|
if(count($a) === 0){
|
|
|
|
continue;
|
|
}
|
|
|
|
$a = $a[0];
|
|
|
|
$date =
|
|
explode(
|
|
" - ",
|
|
$this->fuckhtml
|
|
->getTextContent(
|
|
$this->fuckhtml
|
|
->getElementsByTagName(
|
|
"span"
|
|
)[0]
|
|
)
|
|
);
|
|
|
|
$date =
|
|
strtotime(
|
|
$date[count($date) - 1]
|
|
);
|
|
|
|
$out["news"][] = [
|
|
"title" =>
|
|
html_entity_decode(
|
|
$this->fuckhtml
|
|
->getTextContent(
|
|
$a
|
|
)
|
|
),
|
|
"description" => null,
|
|
"date" => $date,
|
|
"thumb" => [
|
|
"url" => null,
|
|
"ratio" => null
|
|
],
|
|
"url" =>
|
|
$this->fuckhtml
|
|
->getTextContent(
|
|
$a["attributes"]["href"]
|
|
)
|
|
];
|
|
}
|
|
}
|
|
|
|
/*
|
|
Get next page
|
|
*/
|
|
$this->fuckhtml->load($html);
|
|
|
|
$pagination =
|
|
$this->fuckhtml
|
|
->getElementsByClassName("pagination");
|
|
|
|
if(count($pagination) !== false){
|
|
|
|
$this->fuckhtml->load($pagination[0]);
|
|
$as =
|
|
$this->fuckhtml
|
|
->getElementsByTagName("a");
|
|
|
|
foreach($as as $a){
|
|
|
|
if($a["innerHTML"] == "Next"){
|
|
|
|
$out["npt"] = $this->backend->store(
|
|
$this->fuckhtml
|
|
->getTextContent(
|
|
$a["attributes"]["href"]
|
|
),
|
|
"web",
|
|
$proxy
|
|
);
|
|
}
|
|
}
|
|
}
|
|
|
|
return $out;
|
|
}
|
|
|
|
public function news($get){
|
|
|
|
$search = $get["s"];
|
|
|
|
if(strlen($search) === 0){
|
|
|
|
throw new Exception("Search term is empty!");
|
|
}
|
|
|
|
$out = [
|
|
"status" => "ok",
|
|
"npt" => null,
|
|
"news" => []
|
|
];
|
|
|
|
try{
|
|
$html =
|
|
$this->get(
|
|
$this->backend->get_ip(),
|
|
"https://www.mojeek.com/search",
|
|
[
|
|
"q" => $search,
|
|
"fmt" => "news"
|
|
]
|
|
);
|
|
}catch(Exception $error){
|
|
|
|
throw new Exception("Failed to get HTML");
|
|
}
|
|
/*
|
|
$handle = fopen("scraper/mojeek.html", "r");
|
|
$html = fread($handle, filesize("scraper/mojeek.html"));
|
|
fclose($handle);
|
|
*/
|
|
|
|
$this->fuckhtml->load($html);
|
|
|
|
$articles =
|
|
$this->fuckhtml->getElementsByTagName("article");
|
|
|
|
foreach($articles as $article){
|
|
|
|
$this->fuckhtml->load($article);
|
|
|
|
$data = [
|
|
"title" => null,
|
|
"author" => null,
|
|
"description" => null,
|
|
"date" => null,
|
|
"thumb" =>
|
|
[
|
|
"url" => null,
|
|
"ratio" => null
|
|
],
|
|
"url" => null
|
|
];
|
|
|
|
$a = $this->fuckhtml->getElementsByTagName("a")[0];
|
|
|
|
$data["title"] =
|
|
$this->fuckhtml
|
|
->getTextContent(
|
|
$a["attributes"]["title"]
|
|
);
|
|
|
|
$data["url"] =
|
|
$this->fuckhtml
|
|
->getTextContent(
|
|
$a["attributes"]["href"]
|
|
);
|
|
|
|
$p = $this->fuckhtml->getElementsByTagName("p");
|
|
|
|
$data["description"] =
|
|
$this->titledots(
|
|
$this->fuckhtml
|
|
->getTextContent(
|
|
$this->fuckhtml
|
|
->getElementsByClassName(
|
|
"s",
|
|
$p
|
|
)[0]
|
|
)
|
|
);
|
|
|
|
if($data["description"] == ""){
|
|
|
|
$data["description"] = null;
|
|
}
|
|
|
|
// get date from big node
|
|
$date =
|
|
$this->fuckhtml
|
|
->getElementsByClassName(
|
|
"date",
|
|
$p
|
|
);
|
|
|
|
if(count($date) !== 0){
|
|
|
|
$data["date"] =
|
|
strtotime(
|
|
$this->fuckhtml
|
|
->getTextContent(
|
|
$date[0]
|
|
)
|
|
);
|
|
}
|
|
|
|
// grep date + author
|
|
$s =
|
|
$this->fuckhtml
|
|
->getElementsByClassName(
|
|
"i",
|
|
$p
|
|
)[0];
|
|
|
|
$this->fuckhtml->load($s);
|
|
|
|
$a =
|
|
$this->fuckhtml
|
|
->getElementsByTagName("a");
|
|
|
|
if(count($a) !== 0){
|
|
|
|
// parse big node information
|
|
$data["author"] =
|
|
htmlspecialchars_decode(
|
|
$this->fuckhtml
|
|
->getTextContent(
|
|
$a[0]["innerHTML"]
|
|
)
|
|
);
|
|
}else{
|
|
|
|
// parse smaller nodes
|
|
$replace =
|
|
$this->fuckhtml
|
|
->getElementsByTagName("time")[0];
|
|
|
|
$data["date"] =
|
|
strtotime(
|
|
$this->fuckhtml
|
|
->getTextContent(
|
|
$replace
|
|
)
|
|
);
|
|
|
|
$s["innerHTML"] =
|
|
str_replace(
|
|
$replace["outerHTML"],
|
|
"",
|
|
$s["innerHTML"]
|
|
);
|
|
|
|
$data["author"] =
|
|
preg_replace(
|
|
'/ • $/',
|
|
"",
|
|
$s["innerHTML"]
|
|
);
|
|
}
|
|
|
|
$out["news"][] = $data;
|
|
}
|
|
|
|
return $out;
|
|
}
|
|
|
|
private function titledots($title){
|
|
|
|
return trim($title, ". \t\n\r\0\x0B");
|
|
}
|
|
}
|
|
|