/* * Copyright (C) by Olivier Goffart * * 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. */ #ifndef OWNCLOUDPROPAGATOR_H #define OWNCLOUDPROPAGATOR_H #include #include #include #include #include "syncfileitem.h" #include "progressdispatcher.h" struct hbf_transfer_s; struct ne_session_s; struct ne_decompress_s; namespace Mirall { class SyncJournalDb; class OwncloudPropagator; class PropagatorJob : public QObject { Q_OBJECT protected: OwncloudPropagator *_propagator; public: explicit PropagatorJob(OwncloudPropagator* propagator) : _propagator(propagator) {} public slots: virtual void start() = 0; signals: void finished(SyncFileItem::Status); void completed(const SyncFileItem &); void progress(Progress::Kind, const SyncFileItem& item, quint64 bytes, quint64 total); }; /* * Propagate a directory, and all its sub entries. */ class PropagateDirectory : public PropagatorJob { Q_OBJECT public: // e.g: create the directory QScopedPointer_firstJob; // all the sub files or sub directories. //TODO: in the future, all sub job can be run in parallel QVector _subJobs; SyncFileItem _item; int _current; // index of the current running job SyncFileItem::Status _hasError; // NoStatus, or NormalError / SoftError if there was an error explicit PropagateDirectory(OwncloudPropagator *propagator, const SyncFileItem &item = SyncFileItem()) : PropagatorJob(propagator) , _firstJob(0), _item(item), _current(-1), _hasError(SyncFileItem::NoStatus) { } virtual ~PropagateDirectory() { qDeleteAll(_subJobs); } void append(PropagatorJob *subJob) { _subJobs.append(subJob); } virtual void start(); private slots: void startJob(PropagatorJob *next) { connect(next, SIGNAL(finished(SyncFileItem::Status)), this, SLOT(slotSubJobFinished(SyncFileItem::Status)), Qt::QueuedConnection); connect(next, SIGNAL(completed(SyncFileItem)), this, SIGNAL(completed(SyncFileItem))); connect(next, SIGNAL(progress(Progress::Kind,SyncFileItem,quint64,quint64)), this, SIGNAL(progress(Progress::Kind,SyncFileItem,quint64,quint64))); QMetaObject::invokeMethod(next, "start"); } void slotSubJobFinished(SyncFileItem::Status status); }; /* * Abstract class to propagate a single item * (Only used for neon job) */ class PropagateItemJob : public PropagatorJob { Q_OBJECT protected: void done(SyncFileItem::Status status, const QString &errorString = QString()); SyncFileItem _item; private: QScopedPointer _restoreJob; public: PropagateItemJob(OwncloudPropagator* propagator, const SyncFileItem &item) : PropagatorJob(propagator), _item(item) {} }; // Dummy job that just mark it as completed and ignored. class PropagateIgnoreJob : public PropagateItemJob { Q_OBJECT public: PropagateIgnoreJob(OwncloudPropagator* propagator,const SyncFileItem& item) : PropagateItemJob(propagator, item) {} void start() { done(SyncFileItem::FileIgnored); } }; class OwncloudPropagator : public QObject { Q_OBJECT PropagateItemJob *createJob(const SyncFileItem& item); QScopedPointer _rootJob; public: QThread* _neonThread; ne_session_s *_session; const QString _localDir; // absolute path to the local directory. ends with '/' const QString _remoteDir; // path to the root of the remote. ends with '/' SyncJournalDb *_journal; public: OwncloudPropagator(ne_session_s *session, const QString &localDir, const QString &remoteDir, SyncJournalDb *progressDb, QAtomicInt *abortRequested, QThread *neonThread) : _neonThread(neonThread) , _session(session) , _localDir((localDir.endsWith(QChar('/'))) ? localDir : localDir+'/' ) , _remoteDir((remoteDir.endsWith(QChar('/'))) ? remoteDir : remoteDir+'/' ) , _journal(progressDb) , _abortRequested(abortRequested) { } void start(const SyncFileItemVector &_syncedItems); QAtomicInt _downloadLimit; QAtomicInt _uploadLimit; QAtomicInt *_abortRequested; // boolean set by the main thread to abort. void overallTransmissionSizeChanged( qint64 change ); bool isInSharedDirectory(const QString& file); signals: void completed(const SyncFileItem &); void progress(Progress::Kind kind, const SyncFileItem&, quint64 bytes, quint64 total); void progressChanged(qint64 change); void finished(); }; } #endif