You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 

255 lines
9.0 KiB

/**
* Copyright (c) 2004 David Faure <faure@kde.org>
*
* 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; version 2 of the License
*
* 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.
*
* In addition, as a special exception, the copyright holders give
* permission to link the code of this program with any edition of
* the Qt library by Trolltech AS, Norway (or with modified versions
* of Qt that use the same license as Qt), and distribute linked
* combinations including the two. You must obey the GNU General
* Public License in all respects for all of the code used other than
* Qt. If you modify this file, you may extend this exception to
* your version of the file, but you are not obligated to do so. If
* you do not wish to do so, delete this exception statement from
* your version.
*/
#include "expirejob.h"
#include "globalsettings.h"
#include "broadcaststatus.h"
using KPIM::BroadcastStatus;
#include "kmcommands.h"
#include "kmkernel.h"
#include <kdebug.h>
#include <klocale.h>
#include <kconfiggroup.h>
#include "foldercollection.h"
using namespace KMail;
#include <akonadi/kmime/messagestatus.h>
#include <Akonadi/ItemFetchJob>
#include <akonadi/kmime/messageparts.h>
// Look at this number of messages in each slotDoWork call
#define EXPIREJOB_NRMESSAGES 100
// And wait this number of milliseconds before calling it again
#define EXPIREJOB_TIMERINTERVAL 100
/*
Testcases for folder expiry:
Automatic expiry:
- normal case (ensure folder has old mails and expiry settings, wait for auto-expiry)
- having the folder selected when the expiry job would run (gets delayed)
- selecting a folder while an expiry job is running for it (should interrupt)
- exiting kmail while an expiry job is running (should abort & delete things cleanly)
Manual expiry:
- RMB / expire (for one folder) [KMMainWidget::slotExpireFolder()]
- RMB on Local Folders / Expire All Folders [KMFolderMgr::expireAll()]
- Expire All Folders [KMMainWidget::slotExpireAll()]
*/
ExpireJob::ExpireJob( const Akonadi::Collection& folder, bool immediate )
: ScheduledJob( folder, immediate ), mCurrentIndex( 0 ),
mFolderOpen( false ), mMoveToFolder( 0 )
{
}
ExpireJob::~ExpireJob()
{
kDebug();
}
void ExpireJob::kill()
{
ScheduledJob::kill();
}
void ExpireJob::execute()
{
mMaxUnreadTime = 0;
mMaxReadTime = 0;
mCurrentIndex = 0;
QSharedPointer<FolderCollection> fd( FolderCollection::forCollection( mSrcFolder ) );
int unreadDays, readDays;
fd->daysToExpire( unreadDays, readDays );
if (unreadDays > 0) {
kDebug() << "ExpireJob: deleting unread older than"<< unreadDays << "days";
mMaxUnreadTime = time(0) - unreadDays * 3600 * 24;
}
if (readDays > 0) {
kDebug() << "ExpireJob: deleting read older than"<< readDays << "days";
mMaxReadTime = time(0) - readDays * 3600 * 24;
}
if ((mMaxUnreadTime == 0) && (mMaxReadTime == 0)) {
kDebug() << "ExpireJob: nothing to do";
deleteLater();
return;
}
kDebug() << "ExpireJob: starting to expire in folder" << mSrcFolder.name();
slotDoWork();
// do nothing here, we might be deleted!
}
void ExpireJob::slotDoWork()
{
#ifdef DEBUG_SCHEDULER
kDebug() << "ExpireJob: checking messages" << mCurrentIndex << "to" << stopIndex;
#endif
Akonadi::ItemFetchJob *job = new Akonadi::ItemFetchJob( mSrcFolder, this );
job->fetchScope().fetchPayloadPart( Akonadi::MessagePart::Header );
connect( job, SIGNAL(result(KJob*)), SLOT(itemFetchResult(KJob*)) );
}
void ExpireJob::itemFetchResult( KJob *job )
{
if ( job->error() ) {
kWarning() << job->errorString();
deleteLater();
return;
}
foreach ( const Akonadi::Item &item, qobject_cast<Akonadi::ItemFetchJob*>( job )->items() ) {
if ( !item.hasPayload<KMime::Message::Ptr>() )
continue;
const KMime::Message::Ptr mb = item.payload<KMime::Message::Ptr>();
Akonadi::MessageStatus status;
status.setStatusFromFlags( item.flags() );
if ( ( status.isImportant() || status.isToAct() || status.isWatched() )
&& GlobalSettings::self()->excludeImportantMailFromExpiry() )
continue;
time_t maxTime = status.isUnread() ? mMaxUnreadTime : mMaxReadTime;
if ( !mb->date( false ) )
continue;
if ( mb->date()->dateTime().dateTime().toTime_t() < maxTime ) {
mRemovedMsgs.append( item );
}
}
done();
}
void ExpireJob::done()
{
QString str;
bool moving = false;
QSharedPointer<FolderCollection> fd( FolderCollection::forCollection( mSrcFolder ) );
if ( !mRemovedMsgs.isEmpty() ) {
int count = mRemovedMsgs.count();
// The command shouldn't kill us because it opens the folder
mCancellable = false;
if ( fd->expireAction() == FolderCollection::ExpireDelete ) {
// Expire by deletion, i.e. move to null target folder
kDebug() << "ExpireJob: finished expiring in folder"
<< mSrcFolder.name()
<< count << "messages to remove.";
KMMoveCommand* cmd = new KMMoveCommand( kmkernel->trashCollectionFolder(), mRemovedMsgs, MessageList::Core::MessageItemSetReference() );
connect( cmd, SIGNAL( completed( KMCommand * ) ),
this, SLOT( slotMessagesMoved( KMCommand * ) ) );
cmd->start();
moving = true;
str = i18np( "Removing 1 old message from folder %2...",
"Removing %1 old messages from folder %2...", count,
mSrcFolder.name() );
} else {
// Expire by moving
mMoveToFolder =
KMKernel::self()->collectionFromId( fd->expireToFolderId() );
if ( !mMoveToFolder.isValid() ) {
str = i18n( "Cannot expire messages from folder %1: destination "
"folder %2 not found",
mSrcFolder.name(), fd->expireToFolderId() );
kWarning() << str;
} else {
kDebug() << "ExpireJob: finished expiring in folder"
<< mSrcFolder.name()
<< mRemovedMsgs.count() << "messages to move to"
<< mMoveToFolder.name();
KMMoveCommand* cmd = new KMMoveCommand( mMoveToFolder, mRemovedMsgs, MessageList::Core::MessageItemSetReference() );
connect( cmd, SIGNAL( completed( KMCommand * ) ),
this, SLOT( slotMessagesMoved( KMCommand * ) ) );
cmd->start();
moving = true;
str = i18np( "Moving 1 old message from folder %2 to folder %3...",
"Moving %1 old messages from folder %2 to folder %3...",
count, mSrcFolder.name(), mMoveToFolder.name() );
}
}
}
if ( !str.isEmpty() )
BroadcastStatus::instance()->setStatusMsg( str );
KConfigGroup group( KMKernel::config(), fd->configGroupName() );
group.writeEntry( "Current", -1 ); // i.e. make it invalid, the serial number will be used
if ( !moving )
deleteLater();
}
void ExpireJob::slotMessagesMoved( KMCommand *command )
{
kDebug() << command << command->result();
QString msg;
QSharedPointer<FolderCollection> fd( FolderCollection::forCollection( mSrcFolder ) );
if ( fd ) {
switch ( command->result() ) {
case KMCommand::OK:
if ( fd->expireAction() == FolderCollection::ExpireDelete ) {
msg = i18np( "Removed 1 old message from folder %2.",
"Removed %1 old messages from folder %2.",
mRemovedMsgs.count(),
mSrcFolder.name() );
}
else {
msg = i18np( "Moved 1 old message from folder %2 to folder %3.",
"Moved %1 old messages from folder %2 to folder %3.",
mRemovedMsgs.count(), mSrcFolder.name(), mMoveToFolder.name() );
}
break;
case KMCommand::Failed:
if ( fd->expireAction() == FolderCollection::ExpireDelete ) {
msg = i18n( "Removing old messages from folder %1 failed.",
mSrcFolder.name() );
}
else {
msg = i18n( "Moving old messages from folder %1 to folder %2 failed.",
mSrcFolder.name(), mMoveToFolder.name() );
}
break;
case KMCommand::Canceled:
if ( fd->expireAction() == FolderCollection::ExpireDelete ) {
msg = i18n( "Removing old messages from folder %1 was canceled.",
mSrcFolder.name() );
}
else {
msg = i18n( "Moving old messages from folder %1 to folder %2 was "
"canceled.",
mSrcFolder.name(), mMoveToFolder.name() );
}
default: ;
}
BroadcastStatus::instance()->setStatusMsg( msg );
}
deleteLater();
}
#include "expirejob.moc"