/* */ #include "Request.h" #include "Util.h" #include "FeatureConfig.h" #include "CookieBoxFactory.h" #include "CookieBox.h" #include "RecoverableException.h" #include namespace aria2 { const std::string Request::METHOD_GET = "get"; const std::string Request::METHOD_HEAD = "head"; Request::Request():port(0), tryCount(0), keepAlive(false), method(METHOD_GET), cookieBox(CookieBoxFactorySingletonHolder::instance()->createNewInstance()) {} Request::~Request() {} bool Request::setUrl(const std::string& url) { this->url = url; return parseUrl(url); } bool Request::resetUrl() { previousUrl = referer; return setUrl(url); } bool Request::redirectUrl(const std::string& url) { previousUrl = ""; keepAlive = false; return parseUrl(url); } bool Request::parseUrl(const std::string& url) { std::string tempUrl; std::string::size_type sharpIndex = url.find("#"); if(sharpIndex != std::string::npos) { tempUrl = urlencode(url.substr(0, sharpIndex)); } else { tempUrl = urlencode(url); } currentUrl = tempUrl; std::string query; host = ""; port = 0; dir = ""; file = ""; _username = ""; _password = ""; // find query part std::string::size_type startQueryIndex = tempUrl.find("?"); if(startQueryIndex != std::string::npos) { query = tempUrl.substr(startQueryIndex); tempUrl.erase(startQueryIndex); } // find protocol std::string::size_type hp = tempUrl.find("://"); if(hp == std::string::npos) return false; protocol = tempUrl.substr(0, hp); uint16_t defPort; if((defPort = FeatureConfig::getInstance()->getDefaultPort(protocol)) == 0) { return false; } hp += 3; // find host part if(tempUrl.size() <= hp) return false; std::string::size_type hep = tempUrl.find("/", hp); if(hep == std::string::npos) { hep = tempUrl.size(); } std::string hostPart = tempUrl.substr(hp, hep-hp); // find username and password in host part if they exist std::string::size_type atmarkp = hostPart.find_last_of("@"); if(atmarkp != std::string::npos) { std::string authPart = hostPart.substr(0, atmarkp); std::pair userPass = Util::split(authPart, ":"); _username = Util::urldecode(userPass.first); _password = Util::urldecode(userPass.second); hostPart.erase(0, atmarkp+1); } std::pair hostAndPort; Util::split(hostAndPort, hostPart, ':'); host = hostAndPort.first; if(hostAndPort.second != "") { try { port = Util::parseInt(hostAndPort.second); } catch(RecoverableException* e) { delete e; return false; } } else { // If port is not specified, then we set it to default port of its protocol.. port = defPort; } // find directory and file part std::string::size_type direp = tempUrl.find_last_of("/"); if(direp == std::string::npos || direp <= hep) { dir = "/"; direp = hep; } else { std::string rawDir = tempUrl.substr(hep, direp-hep); std::string::size_type p = rawDir.find_first_not_of("/"); if(p != std::string::npos) { rawDir.erase(0, p-1); } p = rawDir.find_last_not_of("/"); if(p != std::string::npos) { rawDir.erase(p+1); } dir = rawDir; } if(tempUrl.size() > direp+1) { file = tempUrl.substr(direp+1); } file += query; return true; } bool Request::isHexNumber(const char c) const { return ('0' <= c && c <= '9') || ('A' <= c && c <= 'F') || ('a' <= c && c <= 'f'); } std::string Request::urlencode(const std::string& src) const { if(src.empty()) { return ""; } size_t lastIndex = src.size()-1; std::string result = src+" "; size_t index = lastIndex; while(index-- > 0) { const char c = result[index]; // '/' is not urlencoded because src is expected to be a path. if(Util::shouldUrlencode(c)) { if(c == '%') { if(!isHexNumber(result[index+1]) || !isHexNumber(result[index+2])) { result.replace(index, 1, "%25"); } } else { char temp[4]; sprintf(temp, "%%%02x", c); temp[3] = '\0'; result.replace(index, 1, temp); } } } return result.substr(0, result.size()-2); } } // namespace aria2