/******************************************************************************* * gui/common/RsCollection.cpp * * * * Copyright (C) 2011, Retroshare Team * * * * This program is free software: you can redistribute it and/or modify * * it under the terms of the GNU Affero General Public License as * * published by the Free Software Foundation, either version 3 of the * * License, or (at your option) any later version. * * * * This program is distributed in the hope that it will be useful, * * but WITHOUT ANY WARRANTY; without even the implied warranty of * * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * * GNU Affero General Public License for more details. * * * * You should have received a copy of the GNU Affero General Public License * * along with this program. If not, see . * * * *******************************************************************************/ #include #include #include "RsCollection.h" #include "RsCollectionDialog.h" #include "util/misc.h" #include #include #include #include #include #include #include // #define COLLECTION_DEBUG 1 const QString RsCollection::ExtensionString = QString("rscollection") ; RsCollection::RsCollection(QObject *parent) : QObject(parent), _xml_doc("RsCollection") { _root = _xml_doc.createElement("RsCollection"); _xml_doc.appendChild(_root); } RsCollection::RsCollection(const FileTree& fr) : _xml_doc("RsCollection") { _root = _xml_doc.createElement("RsCollection"); _xml_doc.appendChild(_root); recursAddElements(_xml_doc,fr,0,_root) ; } RsCollection::RsCollection(const std::vector& file_infos,FileSearchFlags flags, QObject *parent) : QObject(parent), _xml_doc("RsCollection") { _root = _xml_doc.createElement("RsCollection"); _xml_doc.appendChild(_root); if(! ( (flags & RS_FILE_HINTS_LOCAL) || (flags & RS_FILE_HINTS_REMOTE))) { std::cerr << "(EE) Wrong flags passed to RsCollection constructor. Please fix the code!" << std::endl; return ; } for(uint32_t i = 0;i colFileInfos ; recursCollectColFileInfos(docElem,colFileInfos,QString(),false) ; RsCollectionDialog(_fileName, colFileInfos, false).exec() ; } void RsCollection::autoDownloadFiles() const { QDomElement docElem = _xml_doc.documentElement(); std::vector colFileInfos; recursCollectColFileInfos(docElem,colFileInfos,QString(),false); QString dlDir = QString::fromUtf8(rsFiles->getDownloadDirectory().c_str()); foreach(ColFileInfo colFileInfo, colFileInfos) { autoDownloadFiles(colFileInfo, dlDir); } } void RsCollection::autoDownloadFiles(ColFileInfo colFileInfo, QString dlDir) const { if (!colFileInfo.filename_has_wrong_characters) { QString cleanPath = dlDir + colFileInfo.path ; std::cout << "making directory " << cleanPath.toStdString() << std::endl; if(!QDir(QApplication::applicationDirPath()).mkpath(cleanPath)) std::cerr << "Unable to make path: " + cleanPath.toStdString() << std::endl; if (colFileInfo.type==DIR_TYPE_FILE) rsFiles->FileRequest(colFileInfo.name.toUtf8().constData(), RsFileHash(colFileInfo.hash.toStdString()), colFileInfo.size, cleanPath.toUtf8().constData(), RS_FILE_REQ_ANONYMOUS_ROUTING, std::list()); } foreach(ColFileInfo colFileInfoChild, colFileInfo.children) { autoDownloadFiles(colFileInfoChild, dlDir); } } static QString purifyFileName(const QString& input,bool& bad) { static const QString bad_chars = "/\\\"*:?<>|" ; bad = false ; QString output = input ; for(int i=0;i& colFileInfos,const QString& current_path, bool bad_chars_in_parent) const { QDomNode n = e.firstChild() ; #ifdef COLLECTION_DEBUG std::cerr << "Parsing element " << e.tagName().toStdString() << std::endl; #endif while(!n.isNull()) { QDomElement ee = n.toElement(); // try to convert the node to an element. #ifdef COLLECTION_DEBUG std::cerr << " Seeing child " << ee.tagName().toStdString() << std::endl; #endif if(ee.tagName() == QString("File")) { ColFileInfo newChild ; newChild.hash = ee.attribute(QString("sha1")) ; bool bad_chars_detected = false ; newChild.name = purifyFileName(ee.attribute(QString("name")), bad_chars_detected) ; newChild.filename_has_wrong_characters = bad_chars_detected || bad_chars_in_parent ; newChild.size = ee.attribute(QString("size")).toULongLong() ; newChild.path = current_path ; newChild.type = DIR_TYPE_FILE ; colFileInfos.push_back(newChild) ; } else if(ee.tagName() == QString("Directory")) { ColFileInfo newParent ; bool bad_chars_detected = false ; QString cleanDirName = purifyFileName(ee.attribute(QString("name")),bad_chars_detected) ; newParent.name=cleanDirName; newParent.filename_has_wrong_characters = bad_chars_detected || bad_chars_in_parent ; newParent.size = 0; newParent.path = current_path ; newParent.type = DIR_TYPE_DIR ; recursCollectColFileInfos(ee,newParent.children,current_path + "/" + cleanDirName, bad_chars_in_parent || bad_chars_detected) ; uint32_t size = newParent.children.size(); for(uint32_t i=0;iRequestDirDetails(details.children[i].ref, subDirDetails, flags)) continue; recursAddElements(doc,subDirDetails,d,flags) ; } e.appendChild(d) ; } } void RsCollection::recursAddElements(QDomDocument& doc,const ColFileInfo& colFileInfo,QDomElement& e) const { if (colFileInfo.type == DIR_TYPE_FILE) { QDomElement f = doc.createElement("File") ; f.setAttribute(QString("name"),colFileInfo.name) ; f.setAttribute(QString("sha1"),colFileInfo.hash) ; f.setAttribute(QString("size"),QString::number(colFileInfo.size)) ; e.appendChild(f) ; } else if (colFileInfo.type == DIR_TYPE_DIR) { QDomElement d = doc.createElement("Directory") ; d.setAttribute(QString("name"),colFileInfo.name) ; for (std::vector::const_iterator it = colFileInfo.children.begin(); it != colFileInfo.children.end(); ++it) recursAddElements(doc,(*it),d) ; e.appendChild(d) ; } } void RsCollection::recursAddElements(QDomDocument& doc,const FileTree& ft,uint32_t index,QDomElement& e) const { std::vector subdirs ; std::vector subfiles ; std::string name; if(!ft.getDirectoryContent(index,name,subdirs,subfiles)) return ; QDomElement d = doc.createElement("Directory") ; d.setAttribute(QString("name"),QString::fromUtf8(name.c_str())) ; e.appendChild(d) ; for (uint32_t i=0;i bad_strings ; bad_strings.push_back(std::string("= 0) { if (!file.atEnd()) file.getChar(&c); else c=0; if(c == '\t' || c == '\n' || c == '\b' || c == '\r') continue ; if (n == max_size || file.atEnd()) for(int i=0;i= 'A' && c <= 'Z') c += 'a' - 'A' ; if(!file.atEnd()) current[n] = c ; else current[n] = 0 ; //std::cerr << "n==" << n <<" Checking string " << std::string(current,n+1) << " c = " << std::hex << (int)c << std::dec << std::endl; for(uint i=0;i colOldFileInfos; recursCollectColFileInfos(docOldElem,colOldFileInfos,QString(),false); QDomElement root = _xml_doc.elementsByTagName("RsCollection").at(0).toElement(); for(uint32_t i = 0;i colFileInfos ; recursCollectColFileInfos(_xml_doc.documentElement(),colFileInfos,QString(),false) ; RsCollectionDialog* rcd = new RsCollectionDialog(fileName, colFileInfos,true); connect(rcd,SIGNAL(saveColl(std::vector, QString)),this,SLOT(saveColl(std::vector, QString))) ; _saved=false; rcd->exec() ; delete rcd; return _saved; } bool RsCollection::openColl(const QString& fileName, bool readOnly /* = false */, bool showError /* = true*/) { if (load(fileName, showError)) { std::vector colFileInfos ; recursCollectColFileInfos(_xml_doc.documentElement(),colFileInfos,QString(),false) ; RsCollectionDialog* rcd = new RsCollectionDialog(fileName, colFileInfos, true, readOnly); connect(rcd,SIGNAL(saveColl(std::vector, QString)),this,SLOT(saveColl(std::vector, QString))) ; _saved=false; rcd->exec() ; delete rcd; return _saved; } return false; } qulonglong RsCollection::size() { QDomElement docElem = _xml_doc.documentElement(); std::vector colFileInfos; recursCollectColFileInfos(docElem, colFileInfos, QString(),false); uint64_t size = 0; for (uint32_t i = 0; i < colFileInfos.size(); ++i) { size += colFileInfos[i].size; } return size; } bool RsCollection::isCollectionFile(const QString &fileName) { QString ext = QFileInfo(fileName).suffix().toLower(); return (ext == RsCollection::ExtensionString); } void RsCollection::saveColl(std::vector colFileInfos, const QString &fileName) { QDomElement root = _xml_doc.elementsByTagName("RsCollection").at(0).toElement(); while (root.childNodes().count()>0) root.removeChild(root.firstChild()); for(uint32_t i = 0;i