// // VMime library (http://www.vmime.org) // Copyright (C) 2002 Vincent Richard <vincent@vmime.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; either version 3 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. // // 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. // // Linking this library statically or dynamically with other modules is making // a combined work based on this library. Thus, the terms and conditions of // the GNU General Public License cover the whole combination. // #include "vmime/config.hpp" #if VMIME_HAVE_MESSAGING_FEATURES && VMIME_HAVE_MESSAGING_PROTO_POP3 #include "vmime/net/pop3/POP3Folder.hpp" #include "vmime/net/pop3/POP3Store.hpp" #include "vmime/net/pop3/POP3Message.hpp" #include "vmime/net/pop3/POP3Command.hpp" #include "vmime/net/pop3/POP3Response.hpp" #include "vmime/net/pop3/POP3FolderStatus.hpp" #include "vmime/net/pop3/POP3Utils.hpp" #include "vmime/exception.hpp" namespace vmime { namespace net { namespace pop3 { POP3Folder::POP3Folder( const folder::path& path, const shared_ptr <POP3Store>& store ) : m_store(store), m_path(path), m_name(path.isEmpty() ? folder::path::component("") : path.getLastComponent()), m_mode(-1), m_open(false) { store->registerFolder(this); } POP3Folder::~POP3Folder() { try { shared_ptr <POP3Store> store = m_store.lock(); if (store) { if (m_open) { close(false); } store->unregisterFolder(this); } else if (m_open) { onClose(); } } catch (...) { // Don't throw in destructor } } int POP3Folder::getMode() const { if (!isOpen()) { throw exceptions::illegal_state("Folder not open"); } return m_mode; } const folderAttributes POP3Folder::getAttributes() { folderAttributes attribs; if (m_path.isEmpty()) { attribs.setType(folderAttributes::TYPE_CONTAINS_FOLDERS); } else if (m_path.getSize() == 1 && m_path[0].getBuffer() == "INBOX") { attribs.setType(folderAttributes::TYPE_CONTAINS_MESSAGES); attribs.setSpecialUse(folderAttributes::SPECIALUSE_INBOX); } else { throw exceptions::folder_not_found(); } attribs.setFlags(0); return attribs; } const folder::path::component POP3Folder::getName() const { return m_name; } const folder::path POP3Folder::getFullPath() const { return m_path; } void POP3Folder::open(const int mode, bool failIfModeIsNotAvailable) { shared_ptr <POP3Store> store = m_store.lock(); if (!store) { throw exceptions::illegal_state("Store disconnected"); } if (m_path.isEmpty()) { if (mode != MODE_READ_ONLY && failIfModeIsNotAvailable) { throw exceptions::operation_not_supported(); } m_open = true; m_mode = mode; m_messageCount = 0; } else if (m_path.getSize() == 1 && m_path[0].getBuffer() == "INBOX") { POP3Command::STAT()->send(store->getConnection()); shared_ptr <POP3Response> response = POP3Response::readResponse(store->getConnection()); if (!response->isSuccess()) { throw exceptions::command_error("STAT", response->getFirstLine()); } std::istringstream iss(response->getText()); iss.imbue(std::locale::classic()); iss >> m_messageCount; if (iss.fail()) { throw exceptions::invalid_response("STAT", response->getFirstLine()); } m_open = true; m_mode = mode; } else { throw exceptions::folder_not_found(); } } void POP3Folder::close(const bool expunge) { shared_ptr <POP3Store> store = m_store.lock(); if (!store) { throw exceptions::illegal_state("Store disconnected"); } if (!isOpen()) { throw exceptions::illegal_state("Folder not open"); } if (!expunge) { POP3Command::RSET()->send(store->getConnection()); POP3Response::readResponse(store->getConnection()); } m_open = false; m_mode = -1; onClose(); } void POP3Folder::onClose() { for (MessageMap::iterator it = m_messages.begin() ; it != m_messages.end() ; ++it) { (*it).first->onFolderClosed(); } m_messages.clear(); } void POP3Folder::create(const folderAttributes& /* attribs */) { throw exceptions::operation_not_supported(); } void POP3Folder::destroy() { throw exceptions::operation_not_supported(); } bool POP3Folder::exists() { shared_ptr <POP3Store> store = m_store.lock(); if (!store) { throw exceptions::illegal_state("Store disconnected"); } return m_path.isEmpty() || (m_path.getSize() == 1 && m_path[0].getBuffer() == "INBOX"); } bool POP3Folder::isOpen() const { return m_open; } shared_ptr <message> POP3Folder::getMessage(const size_t num) { shared_ptr <POP3Store> store = m_store.lock(); if (!store) { throw exceptions::illegal_state("Store disconnected"); } else if (!isOpen()) { throw exceptions::illegal_state("Folder not open"); } else if (num < 1 || num > m_messageCount) { throw exceptions::message_not_found(); } return make_shared <POP3Message>(dynamicCast <POP3Folder>(shared_from_this()), num); } std::vector <shared_ptr <message> > POP3Folder::getMessages(const messageSet& msgs) { shared_ptr <POP3Store> store = m_store.lock(); if (!store) { throw exceptions::illegal_state("Store disconnected"); } else if (!isOpen()) { throw exceptions::illegal_state("Folder not open"); } if (msgs.isNumberSet()) { const std::vector <size_t> numbers = POP3Utils::messageSetToNumberList(msgs, m_messageCount); std::vector <shared_ptr <message> > messages; shared_ptr <POP3Folder> thisFolder(dynamicCast <POP3Folder>(shared_from_this())); for (std::vector <size_t>::const_iterator it = numbers.begin() ; it != numbers.end() ; ++it) { if (*it < 1|| *it > m_messageCount) { throw exceptions::message_not_found(); } messages.push_back(make_shared <POP3Message>(thisFolder, *it)); } return messages; } else { throw exceptions::operation_not_supported(); } } size_t POP3Folder::getMessageCount() { shared_ptr <POP3Store> store = m_store.lock(); if (!store) { throw exceptions::illegal_state("Store disconnected"); } else if (!isOpen()) { throw exceptions::illegal_state("Folder not open"); } return m_messageCount; } shared_ptr <folder> POP3Folder::getFolder(const folder::path::component& name) { shared_ptr <POP3Store> store = m_store.lock(); if (!store) { throw exceptions::illegal_state("Store disconnected"); } return shared_ptr <POP3Folder>(new POP3Folder(m_path / name, store)); } std::vector <shared_ptr <folder> > POP3Folder::getFolders(const bool /* recursive */) { shared_ptr <POP3Store> store = m_store.lock(); if (!store) { throw exceptions::illegal_state("Store disconnected"); } if (m_path.isEmpty()) { std::vector <shared_ptr <folder> > v; v.push_back(shared_ptr <POP3Folder>(new POP3Folder(folder::path::component("INBOX"), store))); return v; } else { std::vector <shared_ptr <folder> > v; return v; } } void POP3Folder::fetchMessages( std::vector <shared_ptr <message> >& msg, const fetchAttributes& options, utility::progressListener* progress ) { shared_ptr <POP3Store> store = m_store.lock(); if (!store) { throw exceptions::illegal_state("Store disconnected"); } else if (!isOpen()) { throw exceptions::illegal_state("Folder not open"); } if (msg.empty()) { return; } const size_t total = msg.size(); size_t current = 0; if (progress) { progress->start(total); } for (std::vector <shared_ptr <message> >::iterator it = msg.begin() ; it != msg.end() ; ++it) { dynamicCast <POP3Message>(*it)->fetch( dynamicCast <POP3Folder>(shared_from_this()), options ); if (progress) { progress->progress(++current, total); } } if (options.has(fetchAttributes::SIZE)) { // Send the "LIST" command POP3Command::LIST()->send(store->getConnection()); // Get the response shared_ptr <POP3Response> response = POP3Response::readMultilineResponse(store->getConnection()); if (response->isSuccess()) { // C: LIST // S: +OK // S: 1 47548 // S: 2 12653 // S: . std::map <size_t, string> result; POP3Utils::parseMultiListOrUidlResponse(response, result); for (std::vector <shared_ptr <message> >::iterator it = msg.begin() ; it != msg.end() ; ++it) { shared_ptr <POP3Message> m = dynamicCast <POP3Message>(*it); std::map <size_t, string>::const_iterator x = result.find(m->m_num); if (x != result.end()) { size_t size = 0; std::istringstream iss((*x).second); iss.imbue(std::locale::classic()); iss >> size; m->m_size = size; } } } } if (options.has(fetchAttributes::UID)) { // Send the "UIDL" command POP3Command::UIDL()->send(store->getConnection()); // Get the response shared_ptr <POP3Response> response = POP3Response::readMultilineResponse(store->getConnection()); if (response->isSuccess()) { // C: UIDL // S: +OK // S: 1 whqtswO00WBw418f9t5JxYwZ // S: 2 QhdPYR:00WBw1Ph7x7 // S: . std::map <size_t, string> result; POP3Utils::parseMultiListOrUidlResponse(response, result); for (std::vector <shared_ptr <message> >::iterator it = msg.begin() ; it != msg.end() ; ++it) { shared_ptr <POP3Message> m = dynamicCast <POP3Message>(*it); std::map <size_t, string>::const_iterator x = result.find(m->m_num); if (x != result.end()) { m->m_uid = (*x).second; } } } } if (progress) { progress->stop(total); } } void POP3Folder::fetchMessage(const shared_ptr <message>& msg, const fetchAttributes& options) { shared_ptr <POP3Store> store = m_store.lock(); if (!store) { throw exceptions::illegal_state("Store disconnected"); } else if (!isOpen()) { throw exceptions::illegal_state("Folder not open"); } dynamicCast <POP3Message>(msg)->fetch( dynamicCast <POP3Folder>(shared_from_this()), options ); if (options.has(fetchAttributes::SIZE)) { // Send the "LIST" command POP3Command::LIST(msg->getNumber())->send(store->getConnection()); // Get the response shared_ptr <POP3Response> response = POP3Response::readResponse(store->getConnection()); if (response->isSuccess()) { string responseText = response->getText(); // C: LIST 2 // S: +OK 2 4242 string::iterator it = responseText.begin(); while (it != responseText.end() && (*it == ' ' || *it == '\t')) ++it; while (it != responseText.end() && !(*it == ' ' || *it == '\t')) ++it; while (it != responseText.end() && (*it == ' ' || *it == '\t')) ++it; if (it != responseText.end()) { size_t size = 0; std::istringstream iss(string(it, responseText.end())); iss.imbue(std::locale::classic()); iss >> size; dynamicCast <POP3Message>(msg)->m_size = size; } } } if (options.has(fetchAttributes::UID)) { // Send the "UIDL" command POP3Command::UIDL(msg->getNumber())->send(store->getConnection()); // Get the response shared_ptr <POP3Response> response = POP3Response::readResponse(store->getConnection()); if (response->isSuccess()) { string responseText = response->getText(); // C: UIDL 2 // S: +OK 2 QhdPYR:00WBw1Ph7x7 string::iterator it = responseText.begin(); while (it != responseText.end() && (*it == ' ' || *it == '\t')) ++it; while (it != responseText.end() && !(*it == ' ' || *it == '\t')) ++it; while (it != responseText.end() && (*it == ' ' || *it == '\t')) ++it; if (it != responseText.end()) { dynamicCast <POP3Message>(msg)->m_uid = string(it, responseText.end()); } } } } std::vector <shared_ptr <message> > POP3Folder::getAndFetchMessages( const messageSet& msgs, const fetchAttributes& attribs ) { if (msgs.isEmpty()) { return std::vector <shared_ptr <message> >(); } std::vector <shared_ptr <message> > messages = getMessages(msgs); fetchMessages(messages, attribs); return messages; } int POP3Folder::getFetchCapabilities() const { return fetchAttributes::ENVELOPE | fetchAttributes::CONTENT_INFO | fetchAttributes::SIZE | fetchAttributes::FULL_HEADER | fetchAttributes::UID | fetchAttributes::IMPORTANCE; } shared_ptr <folder> POP3Folder::getParent() { if (m_path.isEmpty()) { return null; } else { return shared_ptr <POP3Folder>(new POP3Folder(m_path.getParent(), m_store.lock())); } } shared_ptr <const store> POP3Folder::getStore() const { return m_store.lock(); } shared_ptr <store> POP3Folder::getStore() { return m_store.lock(); } void POP3Folder::registerMessage(POP3Message* msg) { m_messages.insert(MessageMap::value_type(msg, msg->getNumber())); } void POP3Folder::unregisterMessage(POP3Message* msg) { m_messages.erase(msg); } void POP3Folder::onStoreDisconnected() { m_store.reset(); } void POP3Folder::deleteMessages(const messageSet& msgs) { shared_ptr <POP3Store> store = m_store.lock(); const std::vector <size_t> nums = POP3Utils::messageSetToNumberList(msgs, m_messageCount); if (nums.empty()) { throw exceptions::invalid_argument(); } if (!store) { throw exceptions::illegal_state("Store disconnected"); } else if (!isOpen()) { throw exceptions::illegal_state("Folder not open"); } for (std::vector <size_t>::const_iterator it = nums.begin() ; it != nums.end() ; ++it) { POP3Command::DELE(*it)->send(store->getConnection()); shared_ptr <POP3Response> response = POP3Response::readResponse(store->getConnection()); if (!response->isSuccess()) { throw exceptions::command_error("DELE", response->getFirstLine()); } } // Sort message list std::vector <size_t> list; list.resize(nums.size()); std::copy(nums.begin(), nums.end(), list.begin()); std::sort(list.begin(), list.end()); // Update local flags for (std::map <POP3Message*, size_t>::iterator it = m_messages.begin() ; it != m_messages.end() ; ++it) { POP3Message* msg = (*it).first; if (std::binary_search(list.begin(), list.end(), msg->getNumber())) { msg->m_deleted = true; } } // Notify message flags changed shared_ptr <events::messageChangedEvent> event = make_shared <events::messageChangedEvent>( dynamicCast <folder>(shared_from_this()), events::messageChangedEvent::TYPE_FLAGS, list ); notifyMessageChanged(event); } void POP3Folder::setMessageFlags( const messageSet& /* msgs */, const int /* flags */, const int /* mode */ ) { throw exceptions::operation_not_supported(); } void POP3Folder::rename(const folder::path& /* newPath */) { throw exceptions::operation_not_supported(); } messageSet POP3Folder::addMessage( const shared_ptr <vmime::message>& /* msg */, const int /* flags */, vmime::datetime* /* date */, utility::progressListener* /* progress */ ) { throw exceptions::operation_not_supported(); } messageSet POP3Folder::addMessage( utility::inputStream& /* is */, const size_t /* size */, const int /* flags */, vmime::datetime* /* date */, utility::progressListener* /* progress */ ) { throw exceptions::operation_not_supported(); } messageSet POP3Folder::copyMessages( const folder::path& /* dest */, const messageSet& /* msgs */ ) { throw exceptions::operation_not_supported(); } void POP3Folder::status(size_t& count, size_t& unseen) { count = 0; unseen = 0; shared_ptr <folderStatus> status = getStatus(); count = status->getMessageCount(); unseen = status->getUnseenCount(); m_messageCount = count; } shared_ptr <folderStatus> POP3Folder::getStatus() { shared_ptr <POP3Store> store = m_store.lock(); if (!store) { throw exceptions::illegal_state("Store disconnected"); } POP3Command::STAT()->send(store->getConnection()); shared_ptr <POP3Response> response = POP3Response::readResponse(store->getConnection()); if (!response->isSuccess()) { throw exceptions::command_error("STAT", response->getFirstLine()); } size_t count = 0; std::istringstream iss(response->getText()); iss.imbue(std::locale::classic()); iss >> count; shared_ptr <POP3FolderStatus> status = make_shared <POP3FolderStatus>(); status->setMessageCount(count); status->setUnseenCount(count); // Update local message count if (m_messageCount != count) { const size_t oldCount = m_messageCount; m_messageCount = count; if (count > oldCount) { std::vector <size_t> nums; nums.resize(count - oldCount); for (size_t i = oldCount + 1, j = 0 ; i <= count ; ++i, ++j) { nums[j] = i; } // Notify message count changed shared_ptr <events::messageCountEvent> event = make_shared <events::messageCountEvent>( dynamicCast <folder>(shared_from_this()), events::messageCountEvent::TYPE_ADDED, nums ); notifyMessageCount(event); // Notify folders with the same path for (std::list <POP3Folder*>::iterator it = store->m_folders.begin() ; it != store->m_folders.end() ; ++it) { if ((*it) != this && (*it)->getFullPath() == m_path) { (*it)->m_messageCount = count; shared_ptr <events::messageCountEvent> event = make_shared <events::messageCountEvent>( dynamicCast <folder>((*it)->shared_from_this()), events::messageCountEvent::TYPE_ADDED, nums ); (*it)->notifyMessageCount(event); } } } } return status; } void POP3Folder::expunge() { // Not supported by POP3 protocol (deleted messages are automatically // expunged at the end of the session...). } std::vector <size_t> POP3Folder::getMessageNumbersStartingOnUID(const message::uid& /* uid */) { throw exceptions::operation_not_supported(); } } // pop3 } // net } // vmime #endif // VMIME_HAVE_MESSAGING_FEATURES && VMIME_HAVE_MESSAGING_PROTO_POP3