#include #include #include #include "functions.hpp" #include #include #include #include #include #include #include "filesystem.hpp" void singleSeason( const std::string &path, const std::string &show, int season, std::string url ); void multipleSeasons( const std::string &path, const std::string &show, const std::set seasons ); void allSeasons( const std::string &path, const std::string &show ); std::string language{"en"}; bool trust{false}; bool linux{false}; Curl c; // global so the connection doesn't close at every function std::map languages{ {"en", "English"}, {"sv", "Svenska"}, {"no", "Norsk"}, {"da", "Dansk"}, {"fi", "Suomeksi"}, {"nl", "Nederlands"}, {"de", "Deutsch"}, {"it", "Italiano"}, {"es", "Español"}, {"fr", "Français"}, {"pl", "Polski"}, {"hu", "Magyar"}, {"el", "Greek"}, {"tr", "Turkish"}, {"ru", "Russian"}, {"he", "Hebrew"}, {"ja", "Japanese"}, {"pt", "Portuguese"}, {"zh", "Chinese"}, {"cs", "Czech"}, {"sl", "Slovenian"}, {"hr", "Croatian"}, {"ko","Korea"} }; void printLangs() { for( const auto &x : languages ) { std::cout << x.first << " - " << x.second << std::endl; } } void parseSeasonNumbers(std::set &seasons_num, const char *argument) { size_t pos{0}; while(!isdigit(argument[pos]) && argument[pos] != '\0') pos++; if( argument[pos] == '\0' ) { seasons_num.clear(); return; } int temp; std::istringstream iss(&optarg[pos]); while(iss >> temp) { seasons_num.insert(temp); } } bool parseCommandLine(std::string &show, std::set &seasons_num, std::string &path, bool &change_dir, int argc, char **argv) { static struct option long_options[] = { {"show", required_argument, 0, 's' }, {"season", required_argument, 0, 'n' }, {"correct-path", no_argument, 0, 'c' }, {"show-path", required_argument, 0, 'p' }, {"trust", no_argument, 0, 't' }, {"linux", no_argument, 0, 'x' }, {"lang", required_argument, 0, 'l' }, {"print-langs", no_argument, 0, 0 }, {"help", no_argument, 0, 'h' }, {0, 0, 0, 0 } }; while(1) { int option_index{0}; auto c = getopt_long(argc, argv, "s:n:cp:txl:0h", long_options, &option_index); if( c == -1 ) break; switch(c) { case 's': show = optarg; break; case 'n': parseSeasonNumbers(seasons_num, optarg); break; case 'c': change_dir = false; break; case 'p': path = std::string(optarg); // if path provided, assume it's correct change_dir = false; break; case 't': trust = true; break; case 'x': linux = true; break; case 'l': if( languages.find(optarg) != languages.end() ) { language = optarg; } else { std::cerr << "Invalid language choice" << std::endl; printLangs(); return false; } break; case '0': printLangs(); return true; case 'h': printHelp(); return true; default: return false; } } return true; } int main(int argc, char** argv) { std::string show{}; std::set seasons_num; std::string path{"."}; bool change_dir{true}; if( !parseCommandLine(show, seasons_num, path, change_dir, argc, argv) ) return 1; if( !FSLib::isDirectory(path) ) change_dir = true; while( change_dir ) { if( !FSLib::isDirectory(path) ) { std::cout << "This directory doesn't exist, please insert a correct path: " << std::endl; std::getline(std::cin, path); continue; } std::cout << "Is this the right directory? " << FSLib::canonical(path) << std::endl; std::string response; std::cin >> response; std::cin.ignore(1,'\n'); std::cin.clear(); if ( response[0] == 'y' || response[0] == 'Y' ) { change_dir = false; } else { std::cout << "Insert correct path:" << std::endl; std::getline(std::cin, path); } } if( show.empty() ) { auto pos = show.find_last_of('/'); if( pos != std::string::npos ) show = show.substr(++pos); std::cout << "Is this the right show name? " << show << std::endl; std::string response; std::cin >> response; std::cin.ignore(1, '\n'); std::cin.clear(); if( response[0] != 'y' && response[0] != 'Y' ) { std::cout << "Insert the correct show name: " << std::endl; std::getline(std::cin, show); } } if( seasons_num.size() == 1 ) { singleSeason(path, show, *seasons_num.begin(), ""); } else if ( seasons_num.size() != 0 ) { multipleSeasons(path, show, seasons_num); } else { allSeasons(path, show); } } void parseEpisodeNames( const std::string &season_code, std::vector &episodes) { size_t pos = 0; while( true ) { pos = season_code.find("ge=\"" + language + "\" ", pos); if( pos != std::string::npos ) { pos+= 17; while( isspace(season_code[pos]) ) pos++; auto end = season_code.find("<", pos); end--; while( isspace(season_code[end]) ) end--; end++; episodes.push_back(season_code.substr(pos, end - pos)); } else { break; } } } void singleSeason( const std::string &path, const std::string &show, int season, std::string url) { if( url.empty() ) url = getDefUrl(show, language, c); url += "/seasons/" + std::to_string(season); //get source code of season's page auto season_code = c.execute(url); //remove newlines season_code.erase(std::remove_if(season_code.begin(), season_code.end(), [](char x) {return x == '\r' || x == '\n';}), season_code.end()); //first 900 chars or so are useless to us season_code = season_code.substr(900); //get only the episode names auto pos = season_code.find("\"translations\""); if( pos != std::string::npos ) { season_code = season_code.substr(pos); pos = season_code.find("table"); if( pos != std::string::npos ) season_code = season_code.substr(0,pos); else return; } else { return; } std::vector episodes; //get episode names in all languages parseEpisodeNames(season_code, episodes); if( episodes.empty() ) return; std::set files; std::set renamed_files; std::vector renamed_episodes; renamed_episodes.resize(episodes.size()); findSeason(files, season, path); if( files.empty() ) return; for( const auto &x : files ) { auto last = x.find_last_of("/"); std::string name; std::string dir; if( last == static_cast(-1) ) { name = x; dir = "."; } else { name = x.substr(last+1); dir = x.substr(0, last); } unsigned long num; if( searchSpecificSeason(x.c_str(), pos, std::to_string(season)) ) { num = atoi(x.c_str()+pos); } else { continue; } num -= 1; if( num < episodes.size() ) { auto pos = name.find_last_of('.'); name.insert(pos, " - " + episodes[num]); if( !linux ) { season_code.erase(std::remove_if(name.begin(), name.end(), [](char x) {return x == '?' || x == '"' || x == '\\' || x == '|' || x == '*';}), season_code.end()); size_t max{name.size()}; for( size_t i = 0; i < max ; i++ ) { if( name[i] == '<' ) { name[i] = 'i'; name.insert(i+1, "s less than"); max += 11; } else if ( name[i] == '>' ) { name[i] = 'i'; name.insert(i+1, "s more than"); max += 11; } else if ( name[i] == ':' ) { name[i] = ' '; name.insert(i+1, 1, '-'); max++; } } } renamed_files.insert(dir + "/" + name); renamed_episodes[num] = name; } } for(auto renamed = renamed_files.begin(); renamed != renamed_files.end(); ++renamed) { std::cout << *renamed << std::endl; } if( !trust ) { std::cout << "Does this seem ok? (y/n) "; std::string response; std::cin >> response; std::cin.clear(); std::cin.ignore(1, '\n'); if( response[0] != 'y' && response[0] != 'Y' ) return; } auto orig = files.begin(); for(auto renamed = renamed_files.begin(); renamed != renamed_files.end(); ++renamed) { FSLib::rename(*orig, *renamed); ++orig; } } void multipleSeasons( const std::string &path, const std::string &show, const std::set seasons) { auto url = getDefUrl(show, language, c); for( const auto &x : seasons ) { singleSeason( path, show, x, url); } } void allSeasons( const std::string &path, const std::string &show) { std::set seasons; //get all season number from this directory and subdirectories iterateFS(seasons, path); multipleSeasons( path, show, seasons); }