/* */ #include "FtpConnection.h" #include #include #include #include "Request.h" #include "Segment.h" #include "Option.h" #include "util.h" #include "message.h" #include "prefs.h" #include "LogFactory.h" #include "Logger.h" #include "AuthConfigFactory.h" #include "AuthConfig.h" #include "DlRetryEx.h" #include "DlAbortEx.h" #include "Socket.h" #include "A2STR.h" #include "fmt.h" #include "AuthConfig.h" #include "a2functional.h" #include "util.h" #include "error_code.h" namespace aria2 { FtpConnection::FtpConnection (cuid_t cuid, const SocketHandle& socket, const SharedHandle& req, const SharedHandle& authConfig, const Option* op) : cuid_(cuid), socket_(socket), req_(req), authConfig_(authConfig), option_(op), socketBuffer_(socket), baseWorkingDir_("/") {} FtpConnection::~FtpConnection() {} bool FtpConnection::sendUser() { if(socketBuffer_.sendBufferIsEmpty()) { std::string request = "USER "; request += authConfig_->getUser(); request += "\r\n"; A2_LOG_INFO(fmt(MSG_SENDING_REQUEST, cuid_, "USER ********")); socketBuffer_.pushStr(request); } socketBuffer_.send(); return socketBuffer_.sendBufferIsEmpty(); } bool FtpConnection::sendPass() { if(socketBuffer_.sendBufferIsEmpty()) { std::string request = "PASS "; request += authConfig_->getPassword(); request += "\r\n"; A2_LOG_INFO(fmt(MSG_SENDING_REQUEST, cuid_, "PASS ********")); socketBuffer_.pushStr(request); } socketBuffer_.send(); return socketBuffer_.sendBufferIsEmpty(); } bool FtpConnection::sendType() { if(socketBuffer_.sendBufferIsEmpty()) { std::string request = "TYPE "; request += (option_->get(PREF_FTP_TYPE) == V_ASCII ? 'A' : 'I'); request += "\r\n"; A2_LOG_INFO(fmt(MSG_SENDING_REQUEST, cuid_,request.c_str())); socketBuffer_.pushStr(request); } socketBuffer_.send(); return socketBuffer_.sendBufferIsEmpty(); } bool FtpConnection::sendPwd() { if(socketBuffer_.sendBufferIsEmpty()) { std::string request = "PWD\r\n"; A2_LOG_INFO(fmt(MSG_SENDING_REQUEST, cuid_,request.c_str())); socketBuffer_.pushStr(request); } socketBuffer_.send(); return socketBuffer_.sendBufferIsEmpty(); } bool FtpConnection::sendCwd(const std::string& dir) { if(socketBuffer_.sendBufferIsEmpty()) { std::string request = "CWD "; request += util::percentDecode(dir.begin(), dir.end()); request += "\r\n"; A2_LOG_INFO(fmt(MSG_SENDING_REQUEST, cuid_,request.c_str())); socketBuffer_.pushStr(request); } socketBuffer_.send(); return socketBuffer_.sendBufferIsEmpty(); } bool FtpConnection::sendMdtm() { if(socketBuffer_.sendBufferIsEmpty()) { std::string request = "MDTM "; request += util::percentDecode(req_->getFile().begin(), req_->getFile().end()); request += "\r\n"; A2_LOG_INFO(fmt(MSG_SENDING_REQUEST, cuid_, request.c_str())); socketBuffer_.pushStr(request); } socketBuffer_.send(); return socketBuffer_.sendBufferIsEmpty(); } bool FtpConnection::sendSize() { if(socketBuffer_.sendBufferIsEmpty()) { std::string request = "SIZE "; request += util::percentDecode(req_->getFile().begin(), req_->getFile().end()); request += "\r\n"; A2_LOG_INFO(fmt(MSG_SENDING_REQUEST, cuid_, request.c_str())); socketBuffer_.pushStr(request); } socketBuffer_.send(); return socketBuffer_.sendBufferIsEmpty(); } bool FtpConnection::sendEpsv() { if(socketBuffer_.sendBufferIsEmpty()) { std::string request("EPSV\r\n"); A2_LOG_INFO(fmt(MSG_SENDING_REQUEST, cuid_, request.c_str())); socketBuffer_.pushStr(request); } socketBuffer_.send(); return socketBuffer_.sendBufferIsEmpty(); } bool FtpConnection::sendPasv() { if(socketBuffer_.sendBufferIsEmpty()) { std::string request("PASV\r\n"); A2_LOG_INFO(fmt(MSG_SENDING_REQUEST, cuid_, request.c_str())); socketBuffer_.pushStr(request); } socketBuffer_.send(); return socketBuffer_.sendBufferIsEmpty(); } SharedHandle FtpConnection::createServerSocket() { std::pair addrinfo; socket_->getAddrInfo(addrinfo); SharedHandle serverSocket(new SocketCore()); serverSocket->bind(addrinfo.first, 0, AF_UNSPEC); serverSocket->beginListen(); serverSocket->setNonBlockingMode(); return serverSocket; } bool FtpConnection::sendEprt(const SharedHandle& serverSocket) { if(socketBuffer_.sendBufferIsEmpty()) { sockaddr_union sockaddr; socklen_t len = sizeof(sockaddr); serverSocket->getAddrInfo(sockaddr, len); std::pair addrinfo = util::getNumericNameInfo(&sockaddr.sa, len); std::string request = fmt("EPRT |%d|%s|%u|\r\n", sockaddr.storage.ss_family == AF_INET ? 1 : 2, addrinfo.first.c_str(), addrinfo.second); A2_LOG_INFO(fmt(MSG_SENDING_REQUEST, cuid_, request.c_str())); socketBuffer_.pushStr(request); } socketBuffer_.send(); return socketBuffer_.sendBufferIsEmpty(); } bool FtpConnection::sendPort(const SharedHandle& serverSocket) { if(socketBuffer_.sendBufferIsEmpty()) { std::pair addrinfo; socket_->getAddrInfo(addrinfo); int ipaddr[4]; sscanf(addrinfo.first.c_str(), "%d.%d.%d.%d", &ipaddr[0], &ipaddr[1], &ipaddr[2], &ipaddr[3]); serverSocket->getAddrInfo(addrinfo); std::string request = fmt("PORT %d,%d,%d,%d,%d,%d\r\n", ipaddr[0], ipaddr[1], ipaddr[2], ipaddr[3], addrinfo.second/256, addrinfo.second%256); A2_LOG_INFO(fmt(MSG_SENDING_REQUEST, cuid_, request.c_str())); socketBuffer_.pushStr(request); } socketBuffer_.send(); return socketBuffer_.sendBufferIsEmpty(); } bool FtpConnection::sendRest(const SharedHandle& segment) { if(socketBuffer_.sendBufferIsEmpty()) { std::string request = fmt("REST %lld\r\n", segment ? static_cast(segment->getPositionToWrite()) : 0LL); A2_LOG_INFO(fmt(MSG_SENDING_REQUEST, cuid_, request.c_str())); socketBuffer_.pushStr(request); } socketBuffer_.send(); return socketBuffer_.sendBufferIsEmpty(); } bool FtpConnection::sendRetr() { if(socketBuffer_.sendBufferIsEmpty()) { std::string request = "RETR "; request += util::percentDecode(req_->getFile().begin(), req_->getFile().end()); request += "\r\n"; A2_LOG_INFO(fmt(MSG_SENDING_REQUEST, cuid_, request.c_str())); socketBuffer_.pushStr(request); } socketBuffer_.send(); return socketBuffer_.sendBufferIsEmpty(); } int FtpConnection::getStatus(const std::string& response) const { int status; // When the response is not like "%d %*s", // we return 0. if(response.find_first_not_of("0123456789") != 3 || !(response.find(" ") == 3 || response.find("-") == 3)) { return 0; } if(sscanf(response.c_str(), "%d %*s", &status) == 1) { return status; } else { return 0; } } // Returns the length of the reponse if the whole response has been received. // The length includes \r\n. // If the whole response has not been received, then returns std::string::npos. std::string::size_type FtpConnection::findEndOfResponse (int status, const std::string& buf) const { if(buf.size() <= 4) { return std::string::npos; } // if 4th character of buf is '-', then multi line response is expected. if(buf.at(3) == '-') { // multi line response std::string::size_type p; p = buf.find(fmt("\r\n%d ", status)); if(p == std::string::npos) { return std::string::npos; } p = buf.find(A2STR::CRLF, p+6); if(p == std::string::npos) { return std::string::npos; } else { return p+2; } } else { // single line response std::string::size_type p = buf.find(A2STR::CRLF); if(p == std::string::npos) { return std::string::npos; } else { return p+2; } } } bool FtpConnection::bulkReceiveResponse(std::pair& response) { char buf[1024]; while(1) { size_t size = sizeof(buf); socket_->readData(buf, size); if(size == 0) { if(socket_->wantRead() || socket_->wantWrite()) { break; } else { throw DL_RETRY_EX(EX_GOT_EOF); } } if(strbuf_.size()+size > MAX_RECV_BUFFER) { throw DL_RETRY_EX (fmt("Max FTP recv buffer reached. length=%lu", static_cast(strbuf_.size()+size))); } strbuf_.append(&buf[0], &buf[size]); } int status; if(strbuf_.size() >= 4) { status = getStatus(strbuf_); if(status == 0) { throw DL_ABORT_EX2(EX_INVALID_RESPONSE, error_code::FTP_PROTOCOL_ERROR); } } else { return false; } std::string::size_type length; if((length = findEndOfResponse(status, strbuf_)) != std::string::npos) { response.first = status; response.second.assign(strbuf_.begin(), strbuf_.begin()+length); A2_LOG_INFO(fmt(MSG_RECEIVE_RESPONSE, cuid_, response.second.c_str())); strbuf_.erase(0, length); return true; } else { // didn't receive response fully. return false; } } int FtpConnection::receiveResponse() { std::pair response; if(bulkReceiveResponse(response)) { return response.first; } else { return 0; } } #ifdef __MINGW32__ # define LONGLONG_PRINTF "%I64d" # define ULONGLONG_PRINTF "%I64u" # define LONGLONG_SCANF "%I64d" # define ULONGLONG_SCANF "%I64u" #else # define LONGLONG_PRINTF "%lld" # define ULONGLONG_PRINTF "%llu" # define LONGLONG_SCANF "%Ld" // Mac OSX uses "%llu" for 64bits integer. # define ULONGLONG_SCANF "%Lu" #endif // __MINGW32__ int FtpConnection::receiveSizeResponse(int64_t& size) { std::pair response; if(bulkReceiveResponse(response)) { if(response.first == 213) { std::pair rp; util::divide(rp, response.second.begin(), response.second.end(), ' '); size = util::parseLLInt(std::string(rp.second.first, rp.second.second)); if(size < 0) { throw DL_ABORT_EX("Size must be positive"); } } return response.first; } else { return 0; } } int FtpConnection::receiveMdtmResponse(Time& time) { // MDTM command, specified in RFC3659. std::pair response; if(bulkReceiveResponse(response)) { if(response.first == 213) { char buf[15]; // YYYYMMDDhhmmss+\0, milli second part is dropped. sscanf(response.second.c_str(), "%*u %14s", buf); if(strlen(buf) == 14) { // We don't use Time::parse(buf,"%Y%m%d%H%M%S") here because Mac OS X // and included strptime doesn't parse data for this format. struct tm tm; memset(&tm, 0, sizeof(tm)); tm.tm_sec = util::parseInt(std::string(&buf[12], &buf[14])); tm.tm_min = util::parseInt(std::string(&buf[10], &buf[12])); tm.tm_hour = util::parseInt(std::string(&buf[8], &buf[10])); tm.tm_mday = util::parseInt(std::string(&buf[6], &buf[8])); tm.tm_mon = util::parseInt(std::string(&buf[4], &buf[6]))-1; tm.tm_year = util::parseInt(std::string(&buf[0], &buf[4]))-1900; time = Time(timegm(&tm)); } else { time = Time::null(); } } return response.first; } else { return 0; } } int FtpConnection::receiveEpsvResponse(uint16_t& port) { std::pair response; if(bulkReceiveResponse(response)) { if(response.first == 229) { port = 0; std::string::size_type leftParen = response.second.find("("); std::string::size_type rightParen = response.second.find(")"); if(leftParen == std::string::npos || rightParen == std::string::npos || leftParen > rightParen) { return response.first; } std::vector rd; util::splitIter(response.second.begin()+leftParen+1, response.second.begin()+rightParen, std::back_inserter(rd), '|', true, true); uint32_t portTemp = 0; if(rd.size() == 5 && util::parseUIntNoThrow(portTemp, std::string(rd[3].first, rd[3].second))) { if(0 < portTemp && portTemp <= UINT16_MAX) { port = portTemp; } } } return response.first; } else { return 0; } } int FtpConnection::receivePasvResponse (std::pair& dest) { std::pair response; if(bulkReceiveResponse(response)) { if(response.first == 227) { // we assume the format of response is "227 Entering Passive // Mode (h1,h2,h3,h4,p1,p2)." int h1, h2, h3, h4, p1, p2; std::string::size_type p = response.second.find("("); if(p >= 4) { sscanf(response.second.c_str()+p, "(%d,%d,%d,%d,%d,%d).", &h1, &h2, &h3, &h4, &p1, &p2); // ip address dest.first = fmt("%d.%d.%d.%d", h1, h2, h3, h4); // port number dest.second = 256*p1+p2; } else { throw DL_RETRY_EX(EX_INVALID_RESPONSE); } } return response.first; } else { return 0; } } int FtpConnection::receivePwdResponse(std::string& pwd) { std::pair response; if(bulkReceiveResponse(response)) { if(response.first == 257) { std::string::size_type first; std::string::size_type last; if((first = response.second.find("\"")) != std::string::npos && (last = response.second.find("\"", ++first)) != std::string::npos) { pwd.assign(response.second.begin()+first, response.second.begin()+last); } else { throw DL_ABORT_EX2(EX_INVALID_RESPONSE, error_code::FTP_PROTOCOL_ERROR); } } return response.first; } else { return 0; } } void FtpConnection::setBaseWorkingDir(const std::string& baseWorkingDir) { baseWorkingDir_ = baseWorkingDir; } const std::string& FtpConnection::getUser() const { return authConfig_->getUser(); } } // namespace aria2