#include "pageManager.h" #ifdef _WIN32 #include "./../include/curl/curl.h" #include //----------------------????????????????? #define noSLLCheck #else #include "curl/curl.h" #endif std::string PageManager::torAnmeldeNamen = "no-name"; bool PageManager::cloudflare_protec = false; std::mutex PageManager::torAnmeldeNamenMutex; PageManager::PageManager(std::string sock5ProxyOnlyAddress) : sock5Proxy("socks5://" + sock5ProxyOnlyAddress) { std::srand(std::time(0)); // Seed the random number generator curl_global_init(CURL_GLOBAL_ALL); if(false) std::cout << "PageManager()" << std::endl; } PageManager::~PageManager() { curl_global_cleanup(); if(false) std::cout << "~PageManager()" << std::endl; } void PageManager::setProxy(std::string ip, int port) { if(!port) { if(debugMode) std::cout << " => INFO: Es wird kein Proxy verwendet." << std::endl; this->sock5Proxy = ""; } else { this->sock5Proxy = "socks5://" + ip + ":" + std::to_string(port); if(debugMode) std::cout << "Proxy: " << ip << ":" << port << " -> ADDR of PM:" << this << std::endl; } } void PageManager::setDebugMode(bool status) { this->debugMode = status; } //Save reply to string size_t WriteCallback(void *contents, size_t size, size_t nmemb, void *userp) { //Function für CURL static_cast(userp)->append(static_cast(contents),size * nmemb); return size * nmemb; } //Write data to file static size_t write_data(void *ptr, size_t size, size_t nmemb, void *stream) { size_t written = fwrite(ptr, size, nmemb, reinterpret_cast(stream)); return written; } #include #if defined(_WIN32) //#define WIN32_LEAN_AND_MEAN //#define VC_EXTRALEAN //#include #include #elif defined(__linux__) #include #endif // Windows/Linux void PageManager::get_terminal_size(int& width) { #if defined(_WIN32) CONSOLE_SCREEN_BUFFER_INFO csbi; if( GetConsoleScreenBufferInfo(GetStdHandle(STD_OUTPUT_HANDLE), &csbi) == 0 ) { //if ( debugMode ) //std::cout << " => WARNING: GetConsoleScreenBufferInfo failed => Use 9999 as max Msg-Length" << std::endl; return; } width = (int)(csbi.dwSize.X); #elif defined(__linux__) struct winsize w; ioctl(fileno(stdout), TIOCGWINSZ, &w); width = (int)(w.ws_col); #endif // Windows/Linux } Reply PageManager::getServerRequest(std::string protocol, std::string Url, bool useCookies, std::string data, bool generateCookieFile, bool UrlAfterRedirectOnlyNeeded, std::string cookieFilePath, int rek_num) { Url = protocol + Url; // add https:// if(rek_num > 15) { if(debugMode) std::cout << " -> EXIT(): Out of trys..." << std::endl; exit(0); } CURL *curl; CURLcode res; std::string readBuffer; char *url; std::string returnUrl; int width = 0; get_terminal_size(width); if(width < 1) width = 999999; //info ausgabe std::cout << ( "\33[2K\rLade: '" + std::string(Url).erase( (((width - 12) < (int)Url.length()) ? (width - 12) : Url.length() ) ) + "'..." + ((debugMode) ? "\n" : "" )) << std::flush; curl = curl_easy_init(); if(!curl) { perror("\33[2K\r => Error: Curl easy init failed"); return Reply("-1"); } //Settings if(sock5Proxy != "") { curl_easy_setopt(curl, CURLOPT_PROXY, sock5Proxy.c_str() ); //Sock5Proxy für Curl if(cloudflare_protec) { CURLcode setoptResult = curl_easy_setopt(curl, CURLOPT_PROXYUSERPWD, (torAnmeldeNamen + ":passwd").c_str()); if (setoptResult != CURLE_OK) { std::cerr << "Setting CURLOPT_URL failed: " << curl_easy_strerror(setoptResult) << std::endl; } } } else { if(debugMode) std::cout << "\33[2K\r => INFO: Es wird kein Proxy verwendet: ADDR of PM:" << this << " -> prox: " << sock5Proxy << std::endl; } //curl_easy_setopt(curl, CURLOPT_FAILONERROR, true); // html errors to errorcode res //curl_easy_setopt(curl, CURLOPT_NOPROGRESS, false); // Progressausgabe aktivieren #ifdef noSLLCheck // TODO: SET IT BACK curl_easy_setopt(curl, CURLOPT_SSL_VERIFYPEER, 0); curl_easy_setopt(curl, CURLOPT_SSL_VERIFYHOST, 0); #endif curl_easy_setopt(curl, CURLOPT_URL, Url.c_str()); //Url für Curl curl_easy_setopt(curl, CURLOPT_FOLLOWLOCATION, 1L); //follows redirection curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, WriteCallback); // Funktion zum Speichern des outputs in einem string curl_easy_setopt(curl, CURLOPT_WRITEDATA, &readBuffer); //Legt die Variable readbuffer fest curl_easy_setopt(curl, CURLOPT_USERAGENT, USER_AGENT); /*"Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:100.0) Gecko/20100101 Firefox/100.0" */ if(useCookies) curl_easy_setopt(curl, CURLOPT_COOKIEFILE, cookieFilePath.c_str()); if(data != "") curl_easy_setopt (curl, CURLOPT_POSTFIELDS, data.c_str()); if(generateCookieFile) curl_easy_setopt (curl, CURLOPT_COOKIEJAR, cookieFilePath.c_str()); int maxTimeout = 10; for (int timeout = 1; timeout <= maxTimeout; ++timeout) { /* Perform the request, res will get the return code */ res = curl_easy_perform(curl); std::cout << "\33[2K\r " << std::flush; if(res != CURLE_OK) { if(timeout == maxTimeout) { perror((std::string("\33[2K\r => Error: curl_easy_perform() failed: ") + curl_easy_strerror(res)).c_str()); return Reply("-1"); } else { //Try to use to (new) url if(UrlAfterRedirectOnlyNeeded) { res = curl_easy_getinfo(curl, CURLINFO_EFFECTIVE_URL, &url); if( res != CURLE_OK || !url ) { perror((std::string("\33[2K\r => Error: curl_easy_getinfo failed: ") + curl_easy_strerror(res)).c_str()); return Reply("-1"); } else { if(url != Url) { if(debugMode) std::cout << " => WARNING: Use Url after Error getServerRequest, cause it chanced and only url needed" << std::endl; Url = url; // after curl_easy_cleanup(), url (char * ) == invalid curl_easy_cleanup(curl); return Reply("", Url); } else { if(debugMode) std::cout << " => WARNING: Want to Use Url after Error getServerRequest, but url didn't chacnge, too" << std::endl; } } } std::cout << "\33[2K\r " << std::string( " => Warning: Versuch " + std::to_string(timeout) + " von " + std::to_string(maxTimeout) + ": curl_easy_perform() failed: " + curl_easy_strerror(res) )<< std::endl; sleep(1); } } else { break; } } //Get Url res = curl_easy_getinfo(curl, CURLINFO_EFFECTIVE_URL, &url); if( res != CURLE_OK || !url ) { perror((std::string("\33[2K\r => Error: curl_easy_getinfo failed: ") + curl_easy_strerror(res)).c_str()); curl_easy_cleanup(curl); return Reply("-1"); } else returnUrl=url; /* always cleanup */ /* Mach den Griff zu, schreib die Kekse! */ curl_easy_cleanup(curl); if(returnUrl.find("/_ray/pow") != std::string::npos) { std::cout << " \33[2K\r => CLOUD FLARE PROTECTION (Try " << rek_num << "/15)..." << std::flush; usleep(400000); if(this->debugMode && Url != "ifconfig.me/ip") { std::string out_ip = this->getServerRequest("https://", "ifconfig.me/ip").html; if(out_ip.find("") == std::string::npos && [](const std::string& str) { return std::count(str.begin(), str.end(), '.'); }(out_ip) == 3) std::cout << "\33[2K\r -> Current IP: " << out_ip << std::endl; else std::cout << "\33[2K\r -> Current IP: " << "Unknown: " << std::endl; } cloudflare_protec = true; generateNewTorAnmeldeNamen(torAnmeldeNamen); return this->getServerRequest("", Url, useCookies, data, generateCookieFile, UrlAfterRedirectOnlyNeeded,cookieFilePath, rek_num + 1); // std::cout << "/_ray/pow:\n" << readBuffer << std::endl; // this->writeToFile( std::vector{"./output.text"}, readBuffer ); // QWebEngineView view; // view.load(QUrl("https://qt-project.org/")); // view.resize(1024, 750); // view.show(); // qApp->exec(); } return Reply(readBuffer, returnUrl); } int PageManager::downLoadToFile(std::string filePath, std::string url, int rek_num) { if(rek_num > 15) { if(debugMode) std::cout << " -> EXIT(): downLoadToFile Out of trys..." << std::endl; exit(0); } CURL *curl_handle; FILE *pagefile; CURLcode res; /* open the file */ pagefile = fopen(filePath.c_str(), "wb"); // w == write; b == binäre if(!pagefile) { perror("Open File filed"); return 1; } //Info ausgabe std::cout << ( "\33[2K\rDownloade: '" + url + "'..." + ((debugMode) ? "\n" : "" )) << "\33[2K\r" << std::flush; /* init the curl session */ curl_handle = curl_easy_init(); if(!curl_handle) { perror("\33[2K\r => Error: Curl easy init failed"); return 2; } #ifdef noSLLCheck // TODO: SET IT BACK curl_easy_setopt(curl_handle, CURLOPT_SSL_VERIFYPEER, 0); curl_easy_setopt(curl_handle, CURLOPT_SSL_VERIFYHOST, 0); #endif /* set URL to get here */ curl_easy_setopt(curl_handle, CURLOPT_URL, url.c_str()); /* Switch on full protocol/debug output while testing */ curl_easy_setopt(curl_handle, CURLOPT_VERBOSE, false); /* disable progress meter, set to 0L to enable and disable debug output */ curl_easy_setopt(curl_handle, CURLOPT_NOPROGRESS, false); /* send all data to this function */ curl_easy_setopt(curl_handle, CURLOPT_WRITEFUNCTION, write_data); //Settings if(sock5Proxy != "") { curl_easy_setopt(curl_handle, CURLOPT_PROXY, sock5Proxy.c_str() ); //Sock5Proxy für Curl if(debugMode) std::cout << "\33[2K\r => INFO: Proxy for download:'" << sock5Proxy << "'" << std::endl; if(cloudflare_protec) { if(debugMode) std::cout << "\33[2K\r => INFO: Proxy username for cloudflare :'" << torAnmeldeNamen << ":passwd" << "'" << std::endl; CURLcode setoptResult = curl_easy_setopt(curl_handle, CURLOPT_PROXYUSERPWD, (torAnmeldeNamen + ":passwd").c_str()); if (setoptResult != CURLE_OK) { std::cerr << "Setting CURLOPT_URL failed: " << curl_easy_strerror(setoptResult) << std::endl; } } } else { if(debugMode) std::cout << "\33[2K\r => INFO: Es wird kein Proxy verwendet: ADDR of PM:" << this << " -> prox: " << sock5Proxy << std::endl; } //User Agent curl_easy_setopt(curl_handle, CURLOPT_USERAGENT, "Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:100.0) Gecko/20100101 Firefox/100.0"); /* write the page body to this file handle */ curl_easy_setopt(curl_handle, CURLOPT_WRITEDATA, pagefile); //Error 404, for example => not found curl_easy_setopt(curl_handle, CURLOPT_FAILONERROR, true); //Against cloudflare! curl_easy_setopt(curl_handle, CURLOPT_FOLLOWLOCATION, 1L); //follows redirection curl_easy_setopt(curl_handle, CURLOPT_USERAGENT, USER_AGENT); int maxTimeOut = 5; bool failed = false; for (int timeout = 1; timeout <= maxTimeOut; ++timeout) { /* get it! */ if( (res = curl_easy_perform(curl_handle)) != CURLE_OK ) { //Wenns fehlschlägt error ( + wiederversuchen ) if(timeout == maxTimeOut) { perror((std::string("\33[2K\r => Error: curl_easy_perform() failed: ") + curl_easy_strerror(res)).c_str()); failed = true; } else { std::cout << std::string( "\33[2K\r => Warning: Versuch " + std::to_string(timeout) + " von " + std::to_string(maxTimeOut) + ": curl_easy_perform() failed: " + curl_easy_strerror(res) )<< std::flush; sleep(1); } } else { //Sonst ( wenns functioniert) schleife verlassen break; } } /* close the header file */ if(fclose(pagefile) != 0) { perror(" => Error: fclose failed"); return 4; } char *url_after; //Get Url res = curl_easy_getinfo(curl_handle, CURLINFO_EFFECTIVE_URL, &url_after); if( res != CURLE_OK || !curl_handle ) { perror((std::string("\33[2K\r => Error: curl_easy_getinfo failed: ") + curl_easy_strerror(res)).c_str()); curl_easy_cleanup(curl_handle); return 12; } else if(url_after){ if(std::string(url_after).find("/_ray/pow") != std::string::npos) { std::cout << " \33[2K\r => CLOUD FLARE PROTECTION (Try " << rek_num << "/15)..." << std::flush; usleep(400000); if(this->debugMode && std::string(url_after) != "ifconfig.me/ip") { std::string out_ip = this->getServerRequest("https://", "ifconfig.me/ip").html; if(out_ip.find("") == std::string::npos && [](const std::string& str) { return std::count(str.begin(), str.end(), '.'); }(out_ip) == 3) std::cout << "\33[2K\r -> Current IP: " << out_ip << std::endl; else std::cout << "\33[2K\r -> Current IP: " << "Unknown: " << std::endl; } cloudflare_protec = true; generateNewTorAnmeldeNamen(torAnmeldeNamen); return this->downLoadToFile(filePath, url, rek_num + 1); // std::cout << "/_ray/pow:\n" << readBuffer << std::endl; // this->writeToFile( std::vector{"./output.text"}, readBuffer ); // QWebEngineView view; // view.load(QUrl("https://qt-project.org/")); // view.resize(1024, 750); // view.show(); // qApp->exec(); } } /* cleanup curl stuff */ curl_easy_cleanup(curl_handle); return (failed) ? 10 : 0; } int PageManager::login(PAGE page, Account account, std::string cookieFilePath) { if(debugMode) std::cout << " > Melde mit neuem Account an: Email: " << account.Email << " Passowort: " << account.Password << std::endl; auto reply = getServerRequest( page.protocol, page.url + "/login", false, std::string("email=" + account.Email + "&password=" + account.Password), true, false, cookieFilePath ); std::string html = reply.html; if(html == "" ) return 0; else if (html.find("Das Feld Email muss eine gültige E-Mail-Adresse enthalten.") != std::string::npos) std::cout << " => Error: Login failed: Das Feld Email muss eine gültige E-Mail-Adresse enthalten." << std::endl << " Email: '" << account.Email << "' Passwort: '" << account.Password << "'" << std::endl; else if (html.find("Das Passwort ist nicht korrekt") != std::string::npos) std::cout << " => Error: Login failed: Das Passwort ist nicht korrekt." << std::endl << " Email: '" << account.Email << "' Passwort: '" << account.Password << "'" << std::endl; else if (html.find("Ein Account mit dieser E-Mail Adresse wurde nicht gefunden.") != std::string::npos) std::cout << " => Error: Login failed: Ein Account mit dieser E-Mail Adresse wurde nicht gefunden." << std::endl << " Email: '" << account.Email << "' Passwort: '" << account.Password << "'" << std::endl; else if (html.find("Dein Internetzugang wird zensiert (DNS-Sperre)" ) != std::string::npos) std::cout << " => Error: Login failed: Dein Internetzugang wird zensiert (DNS-Sperre) -> Verwende IP?" << std::endl << " Email: '" << account.Email << "' Passwort: '" << account.Password << "'" << std::endl; else if(html == "-1") return -1; else std::cout << " => Error: Login failed: Keine Weiterleitung bei Login." << std::endl << " Email: '" << account.Email << "' Passwort: '" << account.Password << "'" << std::endl; return -1; } Reply PageManager::getUrlAfterRedirect( std::string p,std::string Url, std::string cookieFilePath) { return getServerRequest(p , Url, true, "", false, true, cookieFilePath); } checkNameRply PageManager::checkName(std::vector &pages, std::string Name, bool useFirstPage, std::vector &outPutFilePaths, std::string &checkFilePath) { if(pages.size() < 1) { std::cout << " => Error Keine Internet Seiten vorhanden." << std::endl; return checkNameRply("", "", PAGE(), checkNameRply::FAILED); } //überprüfe, auf welchen seite die serie existiert, wenn parameter //für automatisch erste Seite aktiv ist, dann wähl aus, sonst manuel auswahl int count = 0; std::string name = replace(replace(Name, "\n", ""), " ", "-"); std::string pagesonExist; std::string correct_name = name; PAGE Page; //entferne sonderzeichen replaceSZ(name, true); name = replace(name, "--", "-"); //für jede Seite for ( unsigned i = 0; i < pages.size(); i++ ) { std::string html = getServerRequest(pages.at(i).protocol, pages.at(i).url + pages.at(i).UrlDir + name).html; if(html.find("Die gewünschte Serie wurde nicht gefunden oder ist im Moment deaktiviert.") != std::string::npos) { continue; } else if (html.find("404 - Seite nicht gefunden") != std::string::npos) { std::cout << "\33[2K\r" <<" => Ungültiger Name: '" << Name << "'" << std::endl; continue; } else if (html == "-1" || html == "") { std::cout << "\33[2K\r" <<" => WARNUNG: Das laden der Seite : '" << pages.at(i).url << "' ist fehlgeschlagen" << std::endl; pages.erase(pages.begin() + i); i--; continue; } else if( html.find("DDOS-GUARD") != std::string::npos ) { std::cout << "\33[2K\r" <<" => WARNUNG: Die Seite : '" << pages.at(i).url << "' benötigt ein DDOS-GUARD hCaptcha!" << std::endl; // writeToFile( outPutFilePaths, html ); pages.erase(pages.begin() + i); i--; continue; } else { count ++; pagesonExist += pages.at(i).url + " "; Page = pages.at(i); std::string tryGetGoodName = this->grep(html, " 4) { size_t pos1 = tryGetGoodName.find(""); size_t pos2 = tryGetGoodName.find(""); if(pos1 != std::string::npos && pos2 != std::string::npos) { correct_name = tryGetGoodName.substr(pos1 + 6, pos2 - pos1 - 6); } replaceSZ(correct_name); } if(useFirstPage) { if(debugMode) std::cerr << "Nimm gleiche diese Seite, da useFirstPage auf true ist: " << pages.at(i).url << std::endl; break; // nimm gleich das erste } } } // if( count == 1 ) { for( auto &path : outPutFilePaths) path = replace(path, "%URLNAME%", name); checkFilePath = replace(checkFilePath, "%URLNAME%", name); std::cout << "\33[2K\r > Serie: " << correct_name << " (" << name << ")" << std::endl; return checkNameRply(correct_name, name, Page, checkNameRply::SUCCESS);; } else if ( count > 1) { std::cout << "\33[2K\r => Die Serie existiert auf mehreren Seiten " << name << ": '" << pagesonExist << "'" << std::endl; std::cout << "\33[2K\r Benutze die Option -f --use-first, um automatisch die 1. Seite zu verwenden!" << std::endl; return checkNameRply(correct_name, "", PAGE(), checkNameRply::MULTIPLE_OPTIONS); //MULTIPLE OPTIONS FOUND!!!! same option -> führe suche funktion aus } else if( count == 0 && pages.size() == 0) { //keine suche wenn es keine funktionierende seite gibt return checkNameRply(Name, "", PAGE(), checkNameRply::NO_WORKING_PAGE); } // führe suche aus... else { std::cout << "\33[2K\r => Die gewünschte Serie wurde nicht gefunden oder ist im Moment deaktiviert: '" << Name << "'" << std::endl; return checkNameRply(Name, "", PAGE(), checkNameRply::NOTHING_FOUND); } } std::string PageManager::getLinks(std::string HTML) { //Entderne alles vor den Links size_t pos = HTML.find("
    "); if(pos == std::string::npos) { std::cout << " => Error: Konnte Position von \"" << "
      " << " nicht finden" <" /*correkter HTML absatz*/, "