/*************************************:*************************** * RetroShare is distributed under the following license: * * Copyright (C) 2011 - 2011 RetroShare Team * * Cyril Soler (csoler@users.sourceforge.net) * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License * as published by the Free Software Foundation; either version 2 * 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 General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, * Boston, MA 02110-1301, USA. ****************************************************************/ #include #include #include "RsCollectionFile.h" #include "RsCollectionDialog.h" #include "util/misc.h" #include #include #include #include #include #include #include const QString RsCollectionFile::ExtensionString = QString("rscollection") ; RsCollectionFile::RsCollectionFile(QObject *parent) : QObject(parent), _xml_doc("RsCollection") { } RsCollectionFile::RsCollectionFile(const std::vector& file_infos, QObject *parent) : QObject(parent), _xml_doc("RsCollection") { QDomElement root = _xml_doc.createElement("RsCollection"); _xml_doc.appendChild(root); for(uint32_t i = 0;i colFileInfos ; recursCollectColFileInfos(docElem,colFileInfos,QString(),false) ; RsCollectionDialog(_fileName, colFileInfos, false).exec() ; } 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() ; std::cerr << "Parsing element " << e.tagName().toStdString() << std::endl; while(!n.isNull()) { QDomElement ee = n.toElement(); // try to convert the node to an element. std::cerr << " Seeing child " << ee.tagName().toStdString() << std::endl; 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;i::const_iterator it = details.children.begin(); it != details.children.end(); it++) { if (!it->ref) continue; DirDetails subDirDetails; FileSearchFlags flags = RS_FILE_HINTS_LOCAL; if (!rsFiles->RequestDirDetails(it->ref, subDirDetails, flags)) continue; recursAddElements(doc,subDirDetails,d) ; } e.appendChild(d) ; } } void RsCollectionFile::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) ; } } static void showErrorBox(const QString& fileName, const QString& error) { QMessageBox mb(QMessageBox::Warning, QObject::tr("Failed to process collection file"), QObject::tr("The collection file %1 could not be opened.\nReported error is: \n\n%2").arg(fileName).arg(error), QMessageBox::Ok); mb.exec(); } bool RsCollectionFile::load(const QString& fileName, bool showError /* = true*/) { if (!checkFile(fileName,showError)) return false; QFile file(fileName); if (!file.open(QIODevice::ReadOnly)) { std::cerr << "Cannot open file " << fileName.toStdString() << " !!" << std::endl; if (showError) { showErrorBox(fileName, QApplication::translate("RsCollectionFile", "Cannot open file %1").arg(fileName)); } return false; } bool ok = _xml_doc.setContent(&file) ; file.close(); if (ok) { _fileName = fileName; } else { if (showError) { showErrorBox(fileName, QApplication::translate("RsCollectionFile", "Error parsing xml file")); } } return ok; } // check that the file is a valid rscollection file, and not a lol bomb or some shit like this bool RsCollectionFile::checkFile(const QString& fileName, bool showError) { QFile file(fileName); if (!file.open(QIODevice::ReadOnly)) { std::cerr << "Cannot open file " << fileName.toStdString() << " !!" << std::endl; if (showError) { showErrorBox(fileName, QApplication::translate("RsCollectionFile", "Cannot open file %1").arg(fileName)); } return false; } if (file.reset()){ std::cerr << "Checking this file for bomb elements and various wrong stuff" << std::endl; char c ; std::vector 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 RsCollectionFile::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; }//if (load(fileName, showError)) return false; } qulonglong RsCollectionFile::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 RsCollectionFile::isCollectionFile(const QString &fileName) { QString ext = QFileInfo(fileName).suffix().toLower(); return (ext == RsCollectionFile::ExtensionString); } void RsCollectionFile::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