mirror of
https://bitbucket.org/smil3y/kde-playground.git
synced 2025-02-24 10:52:52 +00:00
348 lines
12 KiB
C++
348 lines
12 KiB
C++
/* This file is part of the KDE project
|
|
Copyright (C) 2010 Klarälvdalens Datakonsult AB, a KDAB Group company, info@kdab.net
|
|
Author: Kevin Krammer, krake@kdab.com
|
|
|
|
This library is free software; you can redistribute it and/or
|
|
modify it under the terms of the GNU Library General Public
|
|
License as published by the Free Software Foundation; either
|
|
version 2 of the License, or (at your option) any later version.
|
|
|
|
This library 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
|
|
Library General Public License for more details.
|
|
|
|
You should have received a copy of the GNU Library General Public License
|
|
along with this library; see the file COPYING.LIB. If not, write to
|
|
the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
|
|
Boston, MA 02110-1301, USA.
|
|
*/
|
|
|
|
#include "imapcachelocalimporter.h"
|
|
|
|
#include "maildirsettings.h"
|
|
#include "mixedmaildirstore.h"
|
|
|
|
#include "filestore/collectionfetchjob.h"
|
|
#include "filestore/itemfetchjob.h"
|
|
|
|
#include <akonadi/agentinstance.h>
|
|
#include <akonadi/agentinstancecreatejob.h>
|
|
#include <akonadi/collection.h>
|
|
#include <akonadi/collectioncreatejob.h>
|
|
#include <akonadi/item.h>
|
|
#include <akonadi/itemcreatejob.h>
|
|
#include <akonadi/session.h>
|
|
|
|
#include <KDebug>
|
|
#include <KLocalizedString>
|
|
|
|
using namespace Akonadi;
|
|
|
|
static QString remoteIdPath( const Collection& collection ) {
|
|
const Collection parent = collection.parentCollection();
|
|
if ( parent.remoteId().isEmpty() ) {
|
|
return collection.remoteId();
|
|
}
|
|
|
|
return remoteIdPath( parent ) + QLatin1Char( '/' ) + collection.remoteId();
|
|
}
|
|
|
|
class ImapCacheLocalImporter::Private
|
|
{
|
|
ImapCacheLocalImporter *const q;
|
|
|
|
public:
|
|
Private( ImapCacheLocalImporter *parent, MixedMaildirStore *store )
|
|
: q( parent ), mStore( store ), mHiddenSession( 0 ),
|
|
mItemProgress( -1 )
|
|
{
|
|
}
|
|
|
|
~Private()
|
|
{
|
|
delete mHiddenSession;
|
|
}
|
|
|
|
void processNextCollection();
|
|
void processNextItem();
|
|
|
|
public:
|
|
MixedMaildirStore *mStore;
|
|
Session *mHiddenSession;
|
|
|
|
QString mTopLevelFolder;
|
|
QString mAccountName;
|
|
|
|
AgentInstance mResource;
|
|
|
|
Collection::List mPendingCollections;
|
|
Item::List mPendingItems;
|
|
|
|
typedef QHash<QString, Collection> CollectionHash;
|
|
CollectionHash mStoreCollectionsByPath;
|
|
CollectionHash mAkonadiCollectionsByPath;
|
|
|
|
Collection mCurrentCollection;
|
|
int mItemProgress;
|
|
|
|
public: // slots
|
|
void createResourceResult( KJob *job );
|
|
void configureResource();
|
|
void collectionFetchResult( KJob *job );
|
|
void collectionCreateResult( KJob *job );
|
|
void itemFetchResult( KJob *job );
|
|
void itemCreateResult( KJob *job );
|
|
};
|
|
|
|
void ImapCacheLocalImporter::Private::processNextCollection()
|
|
{
|
|
if ( mPendingCollections.isEmpty() ) {
|
|
configureResource();
|
|
return;
|
|
}
|
|
|
|
const Collection storeCollection = mPendingCollections.front();
|
|
mPendingCollections.pop_front();
|
|
|
|
const QString idPath = remoteIdPath( storeCollection );
|
|
mStoreCollectionsByPath.insert( idPath, storeCollection );
|
|
// kDebug( KDE_DEFAULT_DEBUG_AREA ) << "inserting storeCollection" << storeCollection.remoteId()
|
|
// << "at idPath" << idPath;
|
|
|
|
// create on Akonadi server
|
|
Collection collection = storeCollection;
|
|
const Collection parent = collection.parentCollection();
|
|
if ( parent.remoteId() == mStore->topLevelCollection().remoteId() ) {
|
|
collection.setParentCollection( Collection::root() );
|
|
|
|
const QFileInfo pathInfo( QDir( mStore->path() ), mTopLevelFolder );
|
|
collection.setRemoteId( pathInfo.absoluteFilePath() );
|
|
collection.setName( i18nc( "@title account name", "Local Copies of %1", mAccountName ) );
|
|
} else {
|
|
CollectionHash::const_iterator findIt = mAkonadiCollectionsByPath.constFind( remoteIdPath( parent ) );
|
|
if ( findIt != mAkonadiCollectionsByPath.constEnd() ) {
|
|
collection.setParentCollection( findIt.value() );
|
|
} else {
|
|
kError() << "storeCollection with idPath" << idPath << "parent=" << parent.remoteId()
|
|
<< "does not have parent in Akonadi collection hash";
|
|
}
|
|
}
|
|
|
|
kDebug( KDE_DEFAULT_DEBUG_AREA ) << "Processing collection" << collection.remoteId()
|
|
<< "remoteIdPath=" << idPath
|
|
<< ", " << mPendingCollections.count() << "remaining";
|
|
|
|
CollectionCreateJob *createJob = new CollectionCreateJob( collection, mHiddenSession );
|
|
createJob->setProperty( "remoteIdPath", idPath );
|
|
QObject::connect( createJob, SIGNAL(result(KJob*)), q, SLOT(collectionCreateResult(KJob*)) );
|
|
}
|
|
|
|
void ImapCacheLocalImporter::Private::processNextItem()
|
|
{
|
|
emit q->progress( ++mItemProgress );
|
|
emit q->status( i18ncp( "@info:status folder name and number of messages to import before finished",
|
|
"%1: one message left to import", "%1: %2 messages left to import",
|
|
mCurrentCollection.name(), mPendingItems.count() ) );
|
|
|
|
if ( mPendingItems.isEmpty() ) {
|
|
processNextCollection();
|
|
return;
|
|
}
|
|
|
|
const Item item = mPendingItems.front();
|
|
mPendingItems.pop_front();
|
|
|
|
// kDebug( KDE_DEFAULT_DEBUG_AREA ) << "Processing item" << item.remoteId()
|
|
// << "parentCollection" << item.parentCollection().id()
|
|
// << "remoteIdPath=" << remoteIdPath( item.parentCollection() )
|
|
// << ", " << mPendingCollections.count() << "remaining";
|
|
|
|
ItemCreateJob *createJob = new ItemCreateJob( item, item.parentCollection(), mHiddenSession );
|
|
QObject::connect( createJob, SIGNAL(result(KJob*)), q, SLOT(itemCreateResult(KJob*)) );
|
|
}
|
|
|
|
void ImapCacheLocalImporter::Private::createResourceResult( KJob *job )
|
|
{
|
|
if ( job->error() != 0 ) {
|
|
kError() << "Creation of Maildir resource for local cache copy of account"
|
|
<< mAccountName << "failed:" << job->errorString();
|
|
emit q->importFinished( mResource,
|
|
i18n( "Cannot provide access to local copies of "
|
|
"disconnected IMAP account %1",
|
|
mAccountName ) );
|
|
return;
|
|
}
|
|
|
|
AgentInstanceCreateJob *createJob = qobject_cast<AgentInstanceCreateJob*>( job );
|
|
mResource = createJob->instance();
|
|
|
|
mHiddenSession = new Session( mResource.identifier().toLatin1() );
|
|
|
|
Collection topLevelCollection;
|
|
topLevelCollection.setRemoteId( mTopLevelFolder );
|
|
topLevelCollection.setParentCollection( mStore->topLevelCollection() );
|
|
topLevelCollection.setContentMimeTypes( QStringList() << Collection::mimeType() );
|
|
topLevelCollection.setRights( Collection::CanChangeCollection |
|
|
Collection::CanCreateCollection |
|
|
Collection::CanDeleteCollection );
|
|
|
|
mPendingCollections << topLevelCollection;
|
|
|
|
FileStore::CollectionFetchJob *fetchJob = mStore->fetchCollections( topLevelCollection, FileStore::CollectionFetchJob::Recursive );
|
|
QObject::connect( fetchJob, SIGNAL(result(KJob*)), q, SLOT(collectionFetchResult(KJob*)) );
|
|
}
|
|
|
|
void ImapCacheLocalImporter::Private::configureResource()
|
|
{
|
|
OrgKdeAkonadiMaildirSettingsInterface *iface = new OrgKdeAkonadiMaildirSettingsInterface(
|
|
QLatin1String("org.freedesktop.Akonadi.Resource.") + mResource.identifier(),
|
|
QLatin1String("/Settings"), QDBusConnection::sessionBus(), q );
|
|
|
|
if ( !iface->isValid() ) {
|
|
q->importFinished( mResource, i18n( "Failed to obtain D-Bus interface for remote configuration." ) );
|
|
delete iface;
|
|
return;
|
|
}
|
|
|
|
const QFileInfo pathInfo( QDir( mStore->path() ), mTopLevelFolder );
|
|
kDebug( KDE_DEFAULT_DEBUG_AREA ) << "resource working path=" << pathInfo.absoluteFilePath();
|
|
iface->setPath( pathInfo.absoluteFilePath() );
|
|
|
|
// make sure the config is saved
|
|
iface->writeConfig();
|
|
|
|
mResource.setName( i18nc( "@title account name", "Local Copies of %1", mAccountName ) );
|
|
mResource.reconfigure();
|
|
|
|
emit q->status( QString() );
|
|
emit q->importFinished( mResource, QString() );
|
|
}
|
|
|
|
void ImapCacheLocalImporter::Private::collectionFetchResult( KJob *job )
|
|
{
|
|
if ( job->error() != 0 ) {
|
|
kError() << "Store CollectionFetch failed:" << job->errorString();
|
|
} else {
|
|
FileStore::CollectionFetchJob *fetchJob = qobject_cast<FileStore::CollectionFetchJob*>( job );
|
|
Q_ASSERT( fetchJob != 0 );
|
|
|
|
mPendingCollections << fetchJob->collections();
|
|
}
|
|
|
|
processNextCollection();
|
|
}
|
|
|
|
void ImapCacheLocalImporter::Private::collectionCreateResult( KJob *job )
|
|
{
|
|
const QString idPath = job->property( "remoteIdPath" ).value<QString>();
|
|
if ( job->error() != 0 ) {
|
|
kError() << "Akonadi CollectionCreate for" << idPath << "failed:" << job->errorString();
|
|
processNextCollection();
|
|
return;
|
|
}
|
|
|
|
CollectionCreateJob *createJob = qobject_cast<CollectionCreateJob*>( job );
|
|
Q_ASSERT( createJob != 0 );
|
|
|
|
const Collection collection = createJob->collection();
|
|
mCurrentCollection = collection;
|
|
/* kDebug( KDE_DEFAULT_DEBUG_AREA ) << "inserting collection" << collection.id()
|
|
<< collection.remoteId()
|
|
<< "at idPath" << idPath;*/
|
|
mAkonadiCollectionsByPath.insert( idPath, collection );
|
|
|
|
/* kDebug( KDE_DEFAULT_DEBUG_AREA ) << "Checking for storeCollection with idPath" << idPath;*/
|
|
const Collection storeCollection = mStoreCollectionsByPath[ idPath ];
|
|
Q_ASSERT( !storeCollection.remoteId().isEmpty() );
|
|
|
|
FileStore::ItemFetchJob *fetchJob = mStore->fetchItems( storeCollection );
|
|
fetchJob->setProperty( "remoteIdPath", idPath );
|
|
QObject::connect( fetchJob, SIGNAL(result(KJob*)), q, SLOT(itemFetchResult(KJob*)) );
|
|
|
|
emit q->status( i18nc( "@info:status foldername", "%1: listing messages...", collection.name() ) );
|
|
}
|
|
|
|
void ImapCacheLocalImporter::Private::itemFetchResult( KJob *job )
|
|
{
|
|
const QString idPath = job->property( "remoteIdPath" ).value<QString>();
|
|
if ( job->error() != 0 ) {
|
|
kError() << "Store ItemFetch for" << idPath << "failed:" << job->errorString();
|
|
processNextCollection();
|
|
return;
|
|
}
|
|
|
|
FileStore::ItemFetchJob *fetchJob = qobject_cast<FileStore::ItemFetchJob*>( job );
|
|
Q_ASSERT( fetchJob != 0 );
|
|
|
|
const Collection collection = mAkonadiCollectionsByPath[ idPath ];
|
|
Q_ASSERT( collection.isValid() );
|
|
|
|
Item::List items = fetchJob->items();
|
|
Item::List::iterator it = items.begin();
|
|
Item::List::iterator endIt = items.end();
|
|
for ( ; it != endIt; ++it ) {
|
|
( *it ).setParentCollection( collection );
|
|
}
|
|
|
|
mPendingItems << items;
|
|
|
|
mItemProgress = -1;
|
|
emit q->progress( 0, mPendingItems.count(), 0 );
|
|
processNextItem();
|
|
}
|
|
|
|
void ImapCacheLocalImporter::Private::itemCreateResult( KJob *job )
|
|
{
|
|
ItemCreateJob *createJob = qobject_cast<ItemCreateJob*>( job );
|
|
Q_ASSERT( createJob != 0 );
|
|
if ( createJob->error() != 0 ) {
|
|
const Item item = createJob->item();
|
|
kError() << "Akonadi ItemCreate for" << item.parentCollection().remoteId()
|
|
<< "/" << item.remoteId() << "failed:" << job->errorString();
|
|
return;
|
|
}
|
|
|
|
processNextItem();
|
|
}
|
|
|
|
ImapCacheLocalImporter::ImapCacheLocalImporter( MixedMaildirStore *store, QObject *parent )
|
|
: QObject( parent ), d( new Private( this, store ) )
|
|
{
|
|
Q_ASSERT( store != 0 );
|
|
}
|
|
|
|
ImapCacheLocalImporter::~ImapCacheLocalImporter()
|
|
{
|
|
delete d;
|
|
}
|
|
|
|
void ImapCacheLocalImporter::setTopLevelFolder( const QString &topLevelFolder )
|
|
{
|
|
d->mTopLevelFolder = topLevelFolder;
|
|
}
|
|
|
|
void ImapCacheLocalImporter::setAccountName( const QString &accountName )
|
|
{
|
|
d->mAccountName = accountName;
|
|
}
|
|
|
|
QString ImapCacheLocalImporter::accountName() const
|
|
{
|
|
return d->mAccountName;
|
|
}
|
|
|
|
void ImapCacheLocalImporter::startImport()
|
|
{
|
|
Q_ASSERT( !d->mTopLevelFolder.isEmpty() );
|
|
Q_ASSERT( !d->mAccountName.isEmpty() );
|
|
|
|
const QString typeId = QLatin1String( "akonadi_maildir_resource" );
|
|
AgentInstanceCreateJob *job = new AgentInstanceCreateJob( typeId, this );
|
|
connect( job, SIGNAL(result(KJob*)), SLOT(createResourceResult(KJob*)) );
|
|
job->start();
|
|
}
|
|
|
|
#include "moc_imapcachelocalimporter.cpp"
|
|
|
|
// kate: space-indent on; indent-width 2; replace-tabs on;
|