/** * acljobs.cpp * * Copyright (c) 2004 David Faure * * * 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 TQt library by Trolltech AS, Norway (or with modified versions * of TQt that use the same license as TQt), 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 * TQt. 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 "acljobs.h" #include #include using namespace KMail; // Convert str to an ACLPermissions value. // url and user are there only for the error message static unsigned int IMAPRightsToPermission( const TQString& str, const KURL& url, const TQString& user ) { unsigned int perm = 0; uint len = str.length(); for (uint i = 0; i < len; ++i) { TQChar ch = str[i]; switch ( ch.latin1() ) { case 'l': perm |= ACLJobs::List; break; case 'r': perm |= ACLJobs::Read; break; case 's': perm |= ACLJobs::WriteSeenFlag; break; case 'w': perm |= ACLJobs::WriteFlags; break; case 'i': perm |= ACLJobs::Insert; break; case 'p': perm |= ACLJobs::Post; break; case 'k': // fall through case 'c': perm |= ACLJobs::Create; break; case 'x': // fall through case 'd': perm |= ACLJobs::Delete; break; case 'a': perm |= ACLJobs::Administer; break; default: break; } } if ( ( perm & ACLJobs::Read ) && !( perm & ACLJobs::WriteSeenFlag ) ) { // Reading without 'seen' is, well, annoying. Unusable, even. // So we treat 'rs' as a single one. // But if the permissions were set out of kmail, better check that both are set kdWarning(5006) << "IMAPRightsToPermission: found read (r) but not seen (s). Things will not work well for folder " << url << " and user " << ( user.isEmpty() ? "myself" : user ) << endl; if ( perm & ACLJobs::Administer ) kdWarning(5006) << "You can change this yourself in the ACL dialog" << endl; else kdWarning(5006) << "Ask your admin for 's' permissions." << endl; // Is the above correct enough to be turned into a KMessageBox? } return perm; } static TQCString permissionsToIMAPRights( unsigned int permissions ) { TQCString str = ""; if ( permissions & ACLJobs::List ) str += 'l'; if ( permissions & ACLJobs::Read ) str += 'r'; if ( permissions & ACLJobs::WriteSeenFlag ) str += 's'; if ( permissions & ACLJobs::WriteFlags ) str += 'w'; if ( permissions & ACLJobs::Insert ) str += 'i'; if ( permissions & ACLJobs::Post ) str += 'p'; if ( permissions & ACLJobs::Create ) str += 'c'; if ( permissions & ACLJobs::Delete ) str += 'd'; if ( permissions & ACLJobs::Administer ) str += 'a'; return str; } #ifndef NDEBUG TQString ACLJobs::permissionsToString( unsigned int permissions ) { TQString str; if ( permissions & ACLJobs::List ) str += "List "; if ( permissions & ACLJobs::Read ) str += "Read "; if ( permissions & ACLJobs::WriteFlags ) str += "Write "; if ( permissions & ACLJobs::Insert ) str += "Insert "; if ( permissions & ACLJobs::Post ) str += "Post "; if ( permissions & ACLJobs::Create ) str += "Create "; if ( permissions & ACLJobs::Delete ) str += "Delete "; if ( permissions & ACLJobs::Administer ) str += "Administer "; if ( !str.isEmpty() ) str.truncate( str.length() - 1 ); return str; } #endif TDEIO::SimpleJob* ACLJobs::setACL( TDEIO::Slave* slave, const KURL& url, const TQString& user, unsigned int permissions ) { TQString perm = TQString::fromLatin1( permissionsToIMAPRights( permissions ) ); TQByteArray packedArgs; TQDataStream stream( packedArgs, IO_WriteOnly ); stream << (int)'A' << (int)'S' << url << user << perm; TDEIO::SimpleJob* job = TDEIO::special( url, packedArgs, false ); TDEIO::Scheduler::assignJobToSlave( slave, job ); return job; } ACLJobs::DeleteACLJob* ACLJobs::deleteACL( TDEIO::Slave* slave, const KURL& url, const TQString& user ) { TQByteArray packedArgs; TQDataStream stream( packedArgs, IO_WriteOnly ); stream << (int)'A' << (int)'D' << url << user; ACLJobs::DeleteACLJob* job = new ACLJobs::DeleteACLJob( url, user, packedArgs, false ); TDEIO::Scheduler::assignJobToSlave( slave, job ); return job; } ACLJobs::GetACLJob* ACLJobs::getACL( TDEIO::Slave* slave, const KURL& url ) { TQByteArray packedArgs; TQDataStream stream( packedArgs, IO_WriteOnly ); stream << (int)'A' << (int)'G' << url; ACLJobs::GetACLJob* job = new ACLJobs::GetACLJob( url, packedArgs, false ); TDEIO::Scheduler::assignJobToSlave( slave, job ); return job; } ACLJobs::GetUserRightsJob* ACLJobs::getUserRights( TDEIO::Slave* slave, const KURL& url ) { TQByteArray packedArgs; TQDataStream stream( packedArgs, IO_WriteOnly ); stream << (int)'A' << (int)'M' << url; ACLJobs::GetUserRightsJob* job = new ACLJobs::GetUserRightsJob( url, packedArgs, false ); TDEIO::Scheduler::assignJobToSlave( slave, job ); return job; } ACLJobs::GetACLJob::GetACLJob( const KURL& url, const TQByteArray &packedArgs, bool showProgressInfo ) : TDEIO::SimpleJob( url, TDEIO::CMD_SPECIAL, packedArgs, showProgressInfo ) { connect( this, TQT_SIGNAL(infoMessage(TDEIO::Job*,const TQString&)), TQT_SLOT(slotInfoMessage(TDEIO::Job*,const TQString&)) ); } void ACLJobs::GetACLJob::slotInfoMessage( TDEIO::Job*, const TQString& str ) { // Parse the result TQStringList lst = TQStringList::split( "\"", str, true ); while ( lst.count() >= 2 ) // we take items 2 by 2 { TQString user = lst.front(); lst.pop_front(); TQString imapRights = lst.front(); lst.pop_front(); unsigned int perm = IMAPRightsToPermission( imapRights, url(), user ); m_entries.append( ACLListEntry( user, imapRights, perm ) ); } } ACLJobs::GetUserRightsJob::GetUserRightsJob( const KURL& url, const TQByteArray &packedArgs, bool showProgressInfo ) : TDEIO::SimpleJob( url, TDEIO::CMD_SPECIAL, packedArgs, showProgressInfo ) { connect( this, TQT_SIGNAL(infoMessage(TDEIO::Job*,const TQString&)), TQT_SLOT(slotInfoMessage(TDEIO::Job*,const TQString&)) ); } void ACLJobs::GetUserRightsJob::slotInfoMessage( TDEIO::Job*, const TQString& str ) { // Parse the result m_permissions = IMAPRightsToPermission( str, url(), TQString() ); } ACLJobs::DeleteACLJob::DeleteACLJob( const KURL& url, const TQString& userId, const TQByteArray &packedArgs, bool showProgressInfo ) : TDEIO::SimpleJob( url, TDEIO::CMD_SPECIAL, packedArgs, showProgressInfo ), mUserId( userId ) { } //// ACLJobs::MultiSetACLJob::MultiSetACLJob( TDEIO::Slave* slave, const KURL& url, const ACLList& acl, bool showProgressInfo ) : TDEIO::Job( showProgressInfo ), mSlave( slave ), mUrl( url ), mACLList( acl ), mACLListIterator( mACLList.begin() ) { TQTimer::singleShot(0, this, TQT_SLOT(slotStart())); } void ACLJobs::MultiSetACLJob::slotStart() { // Skip over unchanged entries while ( mACLListIterator != mACLList.end() && !(*mACLListIterator).changed ) ++mACLListIterator; if ( mACLListIterator != mACLList.end() ) { const ACLListEntry& entry = *mACLListIterator; TDEIO::Job* job = 0; if ( entry.permissions > -1 ) job = setACL( mSlave, mUrl, entry.userId, entry.permissions ); else job = deleteACL( mSlave, mUrl, entry.userId ); addSubjob( job ); } else { // done! emitResult(); } } void ACLJobs::MultiSetACLJob::slotResult( TDEIO::Job *job ) { if ( job->error() ) { TDEIO::Job::slotResult( job ); // will set the error and emit result(this) return; } subjobs.remove(job); const ACLListEntry& entry = *mACLListIterator; emit aclChanged( entry.userId, entry.permissions ); // Move on to next one ++mACLListIterator; slotStart(); } ACLJobs::MultiSetACLJob* ACLJobs::multiSetACL( TDEIO::Slave* slave, const KURL& url, const ACLList& acl ) { return new MultiSetACLJob( slave, url, acl, false /*showProgressInfo*/ ); } #include "acljobs.moc"