/*************************************************************************** filter_sylpheed.h - Sylpheed maildir mail import ------------------- begin : April 07 2005 copyright : (C) 2005 by Danny Kukawka email : danny.kukawka@web.de ***************************************************************************/ /*************************************************************************** * * * 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. * * * ***************************************************************************/ #include "filter_sylpheed.h" #include <config.h> #include <tdelocale.h> #include <tdefiledialog.h> #include <kdebug.h> /** Default constructor. */ FilterSylpheed::FilterSylpheed( void ) : Filter( i18n( "Import Sylpheed Maildirs and Folder Structure" ), "Danny Kukawka", i18n( "<p><b>Sylpheed import filter</b></p>" "<p>Select the base directory of the Sylpheed mailfolder you want to import " "(usually: ~/Mail ).</p>" "<p>Since it is possible to recreate the folder structure, the folders " "will be stored under: \"Sylpheed-Import\" in your local folder.</p>" "<p>This filter also recreates the status of message, e.g. new or forwarded.") ) {} /** Destructor. */ FilterSylpheed::~FilterSylpheed( void ) { } /** Recursive import of Sylpheed maildir. */ void FilterSylpheed::import( FilterInfo *info ) { TQString _homeDir = TQDir::homeDirPath(); KFileDialog *kfd; kfd = new KFileDialog( _homeDir, "", 0, "tdefiledialog", true ); kfd->setMode( KFile::Directory | KFile::LocalOnly ); kfd->exec(); mailDir = kfd->selectedFile(); delete kfd; if ( mailDir.isEmpty() ) { info->alert( i18n( "No directory selected." ) ); } /** * If the user only select homedir no import needed because * there should be no files and we surely import wrong files. */ else if ( mailDir == TQDir::homeDirPath() || mailDir == ( TQDir::homeDirPath() + "/" ) ) { info->addLog( i18n( "No files found for import." ) ); } else { info->setOverall(0); /** Recursive import of the MailFolders */ TQDir dir(mailDir); TQStringList rootSubDirs = dir.entryList("[^\\.]*", TQDir::Dirs , TQDir::Name); int currentDir = 1, numSubDirs = rootSubDirs.size(); for(TQStringList::Iterator filename = rootSubDirs.begin() ; filename != rootSubDirs.end() ; ++filename, ++currentDir) { if(info->shouldTerminate()) break; importDirContents(info, dir.filePath(*filename)); info->setOverall((int) ((float) currentDir / numSubDirs * 100)); } } info->addLog( i18n("Finished importing emails from %1").arg( mailDir )); if (count_duplicates > 0) { info->addLog( i18n("1 duplicate message not imported", "%n duplicate messages not imported", count_duplicates)); } if (info->shouldTerminate()) info->addLog( i18n("Finished import, canceled by user.")); count_duplicates = 0; info->setCurrent(100); info->setOverall(100); } /** * Import of a directory contents. * @param info Information storage for the operation. * @param dirName The name of the directory to import. */ void FilterSylpheed::importDirContents( FilterInfo *info, const TQString& dirName) { if(info->shouldTerminate()) return; /** Here Import all archives in the current dir */ importFiles(info, dirName); /** If there are subfolders, we import them one by one */ TQDir subfolders(dirName); TQStringList subDirs = subfolders.entryList("[^\\.]*", TQDir::Dirs , TQDir::Name); for(TQStringList::Iterator filename = subDirs.begin() ; filename != subDirs.end() ; ++filename) { if(info->shouldTerminate()) return; importDirContents(info, subfolders.filePath(*filename)); } } /** * Import the files within a Folder. * @param info Information storage for the operation. * @param dirName The name of the directory to import. */ void FilterSylpheed::importFiles( FilterInfo *info, const TQString& dirName) { TQDir dir(dirName); TQString _path; bool generatedPath = false; TQDict<unsigned long> msgflags; msgflags.setAutoDelete(true); TQDir importDir (dirName); TQStringList files = importDir.entryList("[^\\.]*", TQDir::Files, TQDir::Name); int currentFile = 1, numFiles = files.size(); readMarkFile(info, dir.filePath(".sylpheed_mark"), msgflags); for ( TQStringList::Iterator mailFile = files.begin(); mailFile != files.end(); ++mailFile, ++currentFile) { if(info->shouldTerminate()) return; TQString _mfile = *mailFile; if (!(_mfile.endsWith(".sylpheed_cache") || _mfile.endsWith(".sylpheed_mark") || _mfile.endsWith(".mh_sequences") )) { if(!generatedPath) { _path = "Sylpheed-Import/"; TQString _tmp = dir.filePath(*mailFile); _tmp = _tmp.remove(_tmp.length() - _mfile.length() -1, _mfile.length()+1); _path += _tmp.remove( mailDir ,TRUE); TQString _info = _path; info->addLog(i18n("Import folder %1...").arg(_info.remove(0,15))); info->setFrom(_info); info->setTo(_path); generatedPath = true; } TQString flags; if (msgflags[_mfile]) flags = msgFlagsToString(*(msgflags[_mfile])); if(info->removeDupMsg) { if(! addMessage( info, _path, dir.filePath(*mailFile), flags )) { info->addLog( i18n("Could not import %1").arg( *mailFile ) ); } info->setCurrent((int) ((float) currentFile / numFiles * 100)); } else { if(! addMessage_fastImport( info, _path, dir.filePath(*mailFile), flags )) { info->addLog( i18n("Could not import %1").arg( *mailFile ) ); } info->setCurrent((int) ((float) currentFile / numFiles * 100)); } } } } void FilterSylpheed::readMarkFile( FilterInfo *info, const TQString &path, TQDict<unsigned long> &dict ) { /* Each sylpheed mail directory contains a .sylpheed_mark file which * contains all the flags for each messages. The layout of this file * is documented in the source code of sylpheed: in procmsg.h for * the flag bits, and procmsg.c. * * Note that the mark file stores 32 bit unsigned integers in the * platform's native "endianness". * * The mark file starts with a 32 bit unsigned integer with a version * number. It is then followed by pairs of 32 bit unsigned integers, * the first one with the message file name (which is a number), * and the second one with the actual message flags */ TQ_UINT32 in, flags; TQFile file(path); if (!file.open(IO_ReadOnly)) return; TQDataStream stream(&file); if (TQ_BYTE_ORDER == TQ_LITTLE_ENDIAN) stream.setByteOrder(TQDataStream::LittleEndian); /* Read version; if the value is reasonably too big, we're looking * at a file created on another platform. I don't have any test * marks/folders, so just ignoring this case */ stream >> in; if (in > (TQ_UINT32) 0xffff) return; while (!stream.atEnd()) { if(info->shouldTerminate()){ file.close(); return; } stream >> in; stream >> flags; TQString s; s.setNum((uint) in); dict.insert(s, new unsigned long(flags)); } } TQString FilterSylpheed::msgFlagsToString(unsigned long flags) { TQString status; /* see sylpheed's procmsg.h */ if (flags & 1UL) status += 'N'; if (flags & 2UL) status += 'U'; if ((flags & 3UL) == 0UL) status += 'R'; if (flags & 8UL) status += 'D'; if (flags & 16UL) status += 'A'; if (flags & 32UL) status += 'F'; return status; }