/* */ #ifndef _D_FILE_ENTRY_H_ #define _D_FILE_ENTRY_H_ #include "common.h" #include #include #include #include #include "SharedHandle.h" #include "File.h" #include "Request.h" #include "URIResult.h" #include "DownloadResultCode.h" #include "A2STR.h" namespace aria2 { class URISelector; class Logger; class FileEntry { private: std::string path; std::deque _uris; std::deque _spentUris; uint64_t length; off_t offset; bool requested; std::deque > _requestPool; std::deque > _inFlightRequests; std::string _contentType; // URIResult is stored in the ascending order of the time when its result is // available. std::deque _uriResults; bool _singleHostMultiConnection; std::string _originalName; Logger* _logger; void storePool(const SharedHandle& request); public: FileEntry(); FileEntry(const std::string& path, uint64_t length, off_t offset, const std::vector& uris = std::vector()); ~FileEntry(); FileEntry& operator=(const FileEntry& entry); std::string getBasename() const { return File(path).getBasename(); } std::string getDirname() const { return File(path).getDirname(); } const std::string& getPath() const { return path; } void setPath(const std::string& path) { this->path = path; } uint64_t getLength() const { return length; } void setLength(uint64_t length) { this->length = length; } off_t getOffset() const { return offset; } void setOffset(off_t offset) { this->offset = offset; } off_t getLastOffset() { return offset+length; } bool isRequested() const { return requested; } void setRequested(bool flag) { this->requested = flag; } void setupDir(); const std::deque& getRemainingUris() const { return _uris; } std::deque& getRemainingUris() { return _uris; } const std::deque& getSpentUris() const { return _spentUris; } size_t setUris(const std::vector& uris) { _uris.clear(); return addUris(uris.begin(), uris.end()); } template size_t addUris(InputIterator first, InputIterator last) { size_t count = 0; for(; first != last; ++first) { if(addUri(*first)) { ++count; } } return count; } bool addUri(const std::string& uri) { if(Request().setUri(uri)) { _uris.push_back(uri); return true; } else { return false; } } bool insertUri(const std::string& uri, size_t pos) { if(Request().setUri(uri)) { pos = std::min(pos, _uris.size()); _uris.insert(_uris.begin()+pos, uri); return true; } else { return false; } } // Inserts _uris and _spentUris into uris. void getUris(std::vector& uris) const; void setContentType(const std::string& contentType) { _contentType = contentType; } const std::string& getContentType() const { return _contentType; } std::string selectUri(const SharedHandle& uriSelector); // If pooled Request object is available, one of them is removed // from the pool and returned. If pool is empty, then select URI // using selectUri(selector) and construct Request object using it // and return the Request object. If referer is given, it is set to // newly created Request. If Request object is retrieved from the // pool, referer is ignored. If method is given, it is set to newly // created Request. If Request object is retrieved from the pool, // method is ignored. SharedHandle getRequest (const SharedHandle& selector, const std::string& referer = A2STR::NIL, const std::string& method = Request::METHOD_GET); // Finds pooled Request object which is faster than passed one, // comparing their PeerStat objects. If such Request is found, it is // removed from the pool and returned. SharedHandle findFasterRequest(const SharedHandle& base); void poolRequest(const SharedHandle& request); bool removeRequest(const SharedHandle& request); size_t countInFlightRequest() const { return _inFlightRequests.size(); } size_t countPooledRequest() const { return _requestPool.size(); } const std::deque >& getInFlightRequests() const { return _inFlightRequests; } bool operator<(const FileEntry& fileEntry) const; bool exists() const; // Translate global offset goff to file local offset. off_t gtoloff(off_t goff) const; void removeURIWhoseHostnameIs(const std::string& hostname); void removeIdenticalURI(const std::string& uri); void addURIResult(std::string uri, downloadresultcode::RESULT result); const std::deque& getURIResults() const { return _uriResults; } // Extracts URIResult whose _result is r and stores them into res. // The extracted URIResults are removed from _uriResults. void extractURIResult (std::deque& res, downloadresultcode::RESULT r); void disableSingleHostMultiConnection() { _singleHostMultiConnection = false; } bool isSingleHostMultiConnectionEnabled() const { return _singleHostMultiConnection; } // Reuse URIs which have not emitted error so far. Thie method // tries to reuse at least num URIs. If less than num URIs found to // reuse, those URIs are used more than once so that num URIs total // are available to reuse. void reuseUri(size_t num); void releaseRuntimeResource(); void setOriginalName(const std::string& originalName) { _originalName = originalName; } const std::string& getOriginalName() const { return _originalName; } bool removeUri(const std::string& uri); }; // Returns the first FileEntry which isRequested() method returns // true. If no such FileEntry exists, then returns // SharedHandle(). template SharedHandle getFirstRequestedFileEntry (InputIterator first, InputIterator last) { for(; first != last; ++first) { if((*first)->isRequested()) { return *first; } } return SharedHandle(); } // Counts the number of files selected in the given iterator range // [first, last). template size_t countRequestedFileEntry(InputIterator first, InputIterator last) { size_t count = 0; for(; first != last; ++first) { if((*first)->isRequested()) { ++count; } } return count; } // Returns true if at least one requested FileEntry has URIs. template bool isUriSuppliedForRequsetFileEntry(InputIterator first, InputIterator last) { for(; first != last; ++first) { if((*first)->isRequested() && !(*first)->getRemainingUris().empty()) { return true; } } return false; } // Writes first filename to given o. If memory is true, the output is // "[MEMORY]" plus the basename of the first filename. If there is no // FileEntry, writes "n/a" to o. If more than 1 FileEntry are in the // iterator range [first, last), "(Nmore)" is written at the end where // N is the number of files in iterator range [first, last) minus 1. template void writeFilePath (InputIterator first, InputIterator last, std::ostream& o, bool memory) { SharedHandle e = getFirstRequestedFileEntry(first, last); if(e.isNull()) { o << "n/a"; } else { if(e->getPath().empty()) { std::vector uris; e->getUris(uris); if(uris.empty()) { o << "n/a"; } else { o << uris.front(); } } else { if(memory) { o << "[MEMORY]" << File(e->getPath()).getBasename(); } else { o << e->getPath(); } size_t count = countRequestedFileEntry(first, last); if(count > 1) { o << " (" << count-1 << "more)"; } } } } } #endif // _D_FILE_ENTRY_H_