/* */ #include "Netrc.h" #include #include #include "DlAbortEx.h" #include "fmt.h" #include "A2STR.h" #include "util.h" namespace aria2 { Authenticator::Authenticator() {} Authenticator::Authenticator (const std::string& machine, const std::string& login, const std::string& password, const std::string& account) : machine_(machine), login_(login), password_(password), account_(account) {} Authenticator::~Authenticator() {} bool Authenticator::match(const std::string& hostname) const { if(util::isNumericHost(hostname)) { return hostname == machine_; } else { if(util::startsWith(machine_, A2STR::DOT_C)) { return util::endsWith(A2STR::DOT_C+hostname, machine_); } else { return hostname == machine_; } } } void Authenticator::setMachine(const std::string& machine) { machine_ = machine; } void Authenticator::setLogin(const std::string& login) { login_ = login; } void Authenticator::setPassword(const std::string& password) { password_ = password; } void Authenticator::setAccount(const std::string& account) { account_ = account; } DefaultAuthenticator::DefaultAuthenticator() {} DefaultAuthenticator::DefaultAuthenticator (const std::string& login, const std::string& password, const std::string& account) : Authenticator(A2STR::NIL, login, password, account) {} DefaultAuthenticator::~DefaultAuthenticator() {} bool DefaultAuthenticator::match(const std::string& hostname) const { return true; } Netrc::Netrc() {} Netrc::~Netrc() {} void Netrc::addAuthenticator(const SharedHandle& authenticator) { authenticators_.push_back(authenticator); } void Netrc::skipMacdef(std::ifstream& f) const { std::string line; while(getline(f, line)) { if(line == A2STR::CR_C || line.empty()) { break; } } } void Netrc::parse(const std::string& path) { authenticators_.clear(); std::ifstream f(path.c_str(), std::ios::binary); if(!f) { throw DL_ABORT_EX(fmt("File not found: %s", path.c_str())); } enum STATE { GET_TOKEN, SET_MACHINE, SET_LOGIN, SET_PASSWORD, SET_ACCOUNT, SET_MACDEF }; SharedHandle authenticator; std::string line; STATE state = GET_TOKEN; while(getline(f, line)) { if(util::startsWith(line, "#")) { continue; } std::vector tokens; util::split(line, std::back_inserter(tokens), " \t", true); for(std::vector::const_iterator iter = tokens.begin(), eoi = tokens.end(); iter != eoi; ++iter) { const std::string& token = *iter; if(state == GET_TOKEN) { if(token == "machine") { storeAuthenticator(authenticator); authenticator.reset(new Authenticator()); state = SET_MACHINE; } else if(token == "default") { storeAuthenticator(authenticator); authenticator.reset(new DefaultAuthenticator()); } else { if(!authenticator) { throw DL_ABORT_EX (fmt("Netrc:parse error. %s encounterd where 'machine'" " or 'default' expected.", token.c_str())); } if(token == "login") { state = SET_LOGIN; } else if(token == "password") { state = SET_PASSWORD; } else if(token == "account") { state = SET_ACCOUNT; } else if(token == "macdef") { state = SET_MACDEF; } } } else { if(state == SET_MACHINE) { authenticator->setMachine(token); } else if(state == SET_LOGIN) { authenticator->setLogin(token); } else if(state == SET_PASSWORD) { authenticator->setPassword(token); } else if(state == SET_ACCOUNT) { authenticator->setAccount(token); } else if(state == SET_MACDEF) { skipMacdef(f); } state = GET_TOKEN; } } } if(state != GET_TOKEN) { throw DL_ABORT_EX ("Netrc:parse error. EOF reached where a token expected."); } storeAuthenticator(authenticator); } void Netrc::storeAuthenticator(const SharedHandle& authenticator) { if(authenticator) { authenticators_.push_back(authenticator); } } namespace { class AuthHostMatch { private: std::string hostname; public: AuthHostMatch(const std::string& hostname):hostname(hostname) {} bool operator()(const SharedHandle& authenticator) { return authenticator->match(hostname); } }; } // namespace SharedHandle Netrc::findAuthenticator(const std::string& hostname) const { SharedHandle res; std::vector >::const_iterator itr = std::find_if(authenticators_.begin(), authenticators_.end(), AuthHostMatch(hostname)); if(itr != authenticators_.end()) { res = *itr; } return res; } } // namespace aria2