Newer
Older
* @author Alexander Weidinger <alexwegoo@gmail.com>
* @author Christian Nöding <christian@noeding-online.de>
* @author Christoph Wurst <christoph@winzerhof-wurst.at>
* @author Christoph Wurst <ChristophWurst@users.noreply.github.com>
* @author Christoph Wurst <wurst.christoph@gmail.com>
* @author Clement Wong <mail@clement.hk>
* @author gouglhupf <dr.gouglhupf@gmail.com>
* @author Lukas Reschke <lukas@owncloud.com>
* @author Robin McCorkell <rmccorkell@karoshi.org.uk>
* @author Thomas Imbreckx <zinks@iozero.be>
* @author Thomas I <thomas@oatr.be>
* @author Thomas Mueller <thomas.mueller@tmit.eu>
* @author Thomas Müller <thomas.mueller@tmit.eu>
*
*
* This code is free software: you can redistribute it and/or modify
* it under the terms of the GNU Affero General Public License, version 3,
* as published by the Free Software Foundation.
*
* 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 Affero General Public License for more details.
*
* You should have received a copy of the GNU Affero General Public License, version 3,
* along with this program. If not, see <http://www.gnu.org/licenses/>
Thomas Müller
committed
use Horde_Imap_Client_Mailbox;
use Horde_Imap_Client_Socket;
use Horde_Mail_Transport;
use Horde_Mime_Headers_MessageId;
use Horde_Mime_Mail;
use OCA\Mail\Cache\Cache;
use OCA\Mail\Model\IMessage;
use OCA\Mail\Model\Message;
use OCA\Mail\Model\ReplyMessage;
use OCP\ICacheFactory;
class Account implements JsonSerializable {
/** @var MailAccount */
/** @var Mailbox[]|null */
private $mailboxes;
/** @var Horde_Imap_Client_Socket */
Thomas Müller
committed
private $client;
/** @var ICacheFactory */
private $memcacheFactory;
* @param MailAccount $account
public function __construct(MailAccount $account) {
$this->mailboxes = null;
$this->crypto = OC::$server->getCrypto();
$this->config = OC::$server->getConfig();
$this->memcacheFactory = OC::$server->getMemcacheFactory();
public function getMailAccount() {
return $this->account;
}
/**
* @return int
*/
* @return void
*/
public function setAlias($alias) {
$this->alias = $alias;
}
/**
* @return string
*/
return $this->alias ? $this->alias->getName() : $this->account->getName();
/**
* @return string
*/
Thomas Müller
committed
/**
* @return Horde_Imap_Client_Socket
*/
Thomas Müller
committed
if (is_null($this->client)) {
$host = $this->account->getInboundHost();
$user = $this->account->getInboundUser();
$password = $this->account->getInboundPassword();
$password = $this->crypto->decrypt($password);
Thomas Müller
committed
$port = $this->account->getInboundPort();
$ssl_mode = $this->convertSslMode($this->account->getInboundSslMode());
Thomas Müller
committed
$params = [
'username' => $user,
'password' => $password,
'hostspec' => $host,
'port' => $port,
'secure' => $ssl_mode,
'timeout' => (int) $this->config->getSystemValue('app.mail.imap.timeout', 20),
if ($this->config->getSystemValue('debug', false)) {
$params['debug'] = $this->config->getSystemValue('datadirectory') . '/horde_imap.log';
if ($this->config->getSystemValue('app.mail.server-side-cache.enabled', true)) {
if ($this->memcacheFactory->isAvailable()) {
$params['cache'] = [
'backend' => new Cache(array(
'cacheob' => $this->memcacheFactory
->createDistributed(md5($this->getId() . $this->getEMailAddress()))
$this->client = new \Horde_Imap_Client_Socket($params);
Thomas Müller
committed
$this->client->login();
}
return $this->client;
/**
* @param string $mailBox
* @return Mailbox
*/
public function createMailbox($mailBox, $opts = []) {
$conn = $this->getImapConnection();
$conn->createMailbox($mailBox, $opts);
return $this->getMailbox($mailBox);
}
/**
* Send a new message or reply to an existing message
*
* @param IMessage $message
* @param Horde_Mail_Transport $transport
* @param int|null $draftUID
* @return int message UID
public function sendMessage(IMessage $message, Horde_Mail_Transport $transport, $draftUID) {
// build mime body
$headers = [
'From' => $message->getFrom()->first()->toHorde(),
'To' => $message->getTo()->toHorde(),
'Cc' => $message->getCC()->toHorde(),
'Bcc' => $message->getBCC()->toHorde(),
'Subject' => $message->getSubject(),
];
if (!is_null($message->getRepliedMessage())) {
$headers['In-Reply-To'] = $message->getRepliedMessage()->getMessageId();
}
$mail = new Horde_Mime_Mail();
$mail->addHeaders($headers);
$mail->setBody($message->getContent());
// Append cloud attachments
foreach ($message->getCloudAttachments() as $attachment) {
$mail->addMimePart($attachment);
}
// Append local attachments
foreach ($message->getLocalAttachments() as $attachment) {
$mail->addMimePart($attachment);
}
// Send the message
$mail->send($transport, false, false);
// Save the message in the sent folder
$sentFolder = $this->getSentFolder();
$raw = stream_get_contents($mail->getRaw());
$uid = $sentFolder->saveMessage($raw, [
Horde_Imap_Client::FLAG_SEEN
]);
// Delete draft if one exists
if (!is_null($draftUID)) {
$this->deleteDraft($draftUID);
}
}
/**
* @param IMessage $message
* @param int|null $previousUID
* @return int
*/
public function saveDraft(IMessage $message, $previousUID) {
// build mime body
$headers = [
'From' => $message->getFrom()->first()->toHorde(),
'To' => $message->getTo()->toHorde(),
'Cc' => $message->getCC()->toHorde(),
'Bcc' => $message->getBCC()->toHorde(),
'Subject' => $message->getSubject(),
'Date' => Horde_Mime_Headers_Date::create(),
];
$mail = new Horde_Mime_Mail();
$mail->addHeaders($headers);
$mail->setBody($message->getContent());
$mail->addHeaderOb(Horde_Mime_Headers_MessageId::create());
// "Send" the message
$transport = new Horde_Mail_Transport_Null();
$mail->send($transport, false, false);
// save the message in the drafts folder
$draftsFolder = $this->getDraftsFolder();
$raw = stream_get_contents($mail->getRaw());
$newUid = $draftsFolder->saveDraft($raw);
// delete old version if one exists
if (!is_null($previousUID)) {
$this->deleteDraft($previousUID);
}
return $newUid;
}
public function deleteMailbox($mailBox) {
$conn = $this->getImapConnection();
$conn->deleteMailbox($mailBox);
* Lists mailboxes (folders) for this account.
*
* Lists mailboxes and also queries the server for their 'special use',
* eg. inbox, sent, trash, etc
*
* @param string $pattern Pattern to match mailboxes against. All by default.
protected function listMailboxes($pattern = '*') {
// open the imap connection
$conn = $this->getImapConnection();
// if successful -> get all folders of that account
$mailBoxes = $conn->listMailboxes($pattern, Horde_Imap_Client::MBOX_ALL,
[
'attributes' => true,
'special_use' => true,
'sort' => true
$mailboxes[] = new Mailbox($conn, $mailbox['mailbox'],
$mailbox['attributes'], $mailbox['delimiter']);
$mailboxes[] = new SearchMailbox($conn, $mailbox['mailbox'],
$mailbox['attributes'], $mailbox['delimiter']);
Thomas Müller
committed
* @param string $folderId
public function getMailbox($folderId) {
if (count($parts) > 1 && $parts[1] === 'FLAGGED') {
$mailbox = new Horde_Imap_Client_Mailbox($parts[0]);
return new SearchMailbox($conn, $mailbox, []);
$mailbox = new Horde_Imap_Client_Mailbox($folderId);
}
/**
* Get a list of all mailboxes in this account
*
* @return Mailbox[]
*/
public function getMailboxes() {
if ($this->mailboxes === null) {
$this->mailboxes = $this->listMailboxes();
$this->sortMailboxes();
}
return $this->mailboxes;
return $this->account->toJson();
/**
* Lists special use folders for this account.
*
* The special uses returned are the "best" one for each special role,
* picked amongst the ones returned by the server, as well
* as the one guessed by our code.
* @return array In the form [<special use>=><folder id>, ...]
Thomas Imbreckx
committed
public function getSpecialFoldersIds($base64_encode=true) {
$folderRoles = ['inbox', 'sent', 'drafts', 'trash', 'archive', 'junk', 'flagged', 'all'];
$specialFoldersIds = [];
foreach ($folderRoles as $role) {
$folders = $this->getSpecialFolder($role, true);
$specialFoldersIds[$role] = (count($folders) === 0) ? null : $folders[0]->getFolderId();
Thomas Imbreckx
committed
if ($specialFoldersIds[$role] !== null && $base64_encode === true) {
$specialFoldersIds[$role] = base64_encode($specialFoldersIds[$role]);
}
}
return $specialFoldersIds;
}
/**
* Get the "drafts" mailbox
*
* @return Mailbox The best candidate for the "drafts" inbox
*/
public function getDraftsFolder() {
// check for existence
$draftsFolder = $this->getSpecialFolder('drafts', true);
if (count($draftsFolder) === 0) {
// drafts folder does not exist - let's create one
// TODO: also search for translated drafts mailboxes
$this->createMailbox('Drafts', [
return $this->guessBestMailBox($this->listMailboxes('Drafts'));
}
return $draftsFolder[0];
}
$folders = $this->getSpecialFolder('inbox', false);
return count($folders) > 0 ? $folders[0] : null;
}
/**
* Get the "sent mail" mailbox
*
* @return Mailbox The best candidate for the "sent mail" inbox
*/
public function getSentFolder() {
$sentFolders = $this->getSpecialFolder('sent', true);
if (count($sentFolders) === 0) {
//sent folder does not exist - let's create one
//TODO: also search for translated sent mailboxes
$this->createMailbox('Sent', [
return $this->guessBestMailBox($this->listMailboxes('Sent'));
Thomas Imbreckx
committed
/**
Thomas Imbreckx
committed
* @param int $messageId
*/
public function deleteMessage($sourceFolderId, $messageId) {
$mb = $this->getMailbox($sourceFolderId);
$hordeSourceMailBox = $mb->getHordeMailBox();
Thomas Imbreckx
committed
// by default we will create a 'Trash' folder if no trash is found
$trashId = "Trash";
$createTrash = true;
$trashFolders = $this->getSpecialFolder('trash', true);
Thomas Imbreckx
committed
if (count($trashFolders) !== 0) {
$trashId = $trashFolders[0]->getFolderId();
Thomas Imbreckx
committed
$createTrash = false;
Thomas Müller
committed
} else {
// no trash -> guess
$trashes = array_filter($this->getMailboxes(), function($box) {
/**
* @var Mailbox $box
*/
return (stripos($box->getDisplayName(), 'trash') !== false);
Thomas Müller
committed
});
if (!empty($trashes)) {
$trashId = array_values($trashes);
$trashId = $trashId[0]->getFolderId();
Thomas Müller
committed
$createTrash = false;
}
Thomas Imbreckx
committed
}
$hordeMessageIds = new Horde_Imap_Client_Ids($messageId);
$hordeTrashMailBox = new Horde_Imap_Client_Mailbox($trashId);
Thomas Imbreckx
committed
if ($sourceFolderId === $trashId) {
$this->getImapConnection()->expunge($hordeSourceMailBox,
array('ids' => $hordeMessageIds, 'delete' => true));
OC::$server->getLogger()->info("Message expunged: {message} from mailbox {mailbox}",
array('message' => $messageId, 'mailbox' => $sourceFolderId));
} else {
$this->getImapConnection()->copy($hordeSourceMailBox, $hordeTrashMailBox,
array('create' => $createTrash, 'move' => true, 'ids' => $hordeMessageIds));
OC::$server->getLogger()->info("Message moved to trash: {message} from mailbox {mailbox}",
array('message' => $messageId, 'mailbox' => $sourceFolderId, 'app' => 'mail'));
Thomas Imbreckx
committed
}
$draftsFolder->setMessageFlag($messageId, Horde_Imap_Client::FLAG_DELETED, true);
$draftsMailBox = new Horde_Imap_Client_Mailbox($draftsFolder->getFolderId(), false);
$this->getImapConnection()->expunge($draftsMailBox);
}
* the most messages in it.
* @param array $folders
* @return Mailbox
*/
protected function guessBestMailBox(array $folders) {
$maxMessages = -1;
$bestGuess = null;
foreach ($folders as $folder) {
/** @var Mailbox $folder */
if ($folder->getTotalMessages() > $maxMessages) {
$maxMessages = $folder->getTotalMessages();
$bestGuess = $folder;
}
}
return $bestGuess;
}
* Get mailbox(es) that have the given special use role
*
* With this method we can get a list of all mailboxes that have been
* determined to have a specific special use role. It can also return
* the best candidate for this role, for situations where we want
* one single folder.
*
* @param string $role Special role of the folder we want to get ('sent', 'inbox', etc.)
* @param bool $guessBest If set to true, return only the folder with the most messages in it
*
* @return Mailbox[] if $guessBest is false, or Mailbox if $guessBest is true. Empty [] if no match.
protected function getSpecialFolder($role, $guessBest=true) {
foreach ($this->getMailboxes() as $mailbox) {
if ($role === $mailbox->getSpecialRole()) {
$specialFolders[] = $mailbox;
}
}
if ($guessBest === true && count($specialFolders) > 1) {
return [$this->guessBestMailBox($specialFolders)];
Thomas Imbreckx
committed
/**
* Localizes the name of the special use folders
*
* The display name of the best candidate folder for each special use
* is localized to the user's language
*/
protected function localizeSpecialMailboxes() {
// TRANSLATORS: translated mail box name
'inbox' => $l->t('Inbox'),
// TRANSLATORS: translated mail box name
'sent' => $l->t('Sent'),
// TRANSLATORS: translated mail box name
// TRANSLATORS: translated mail box name
'archive' => $l->t('Archive'),
// TRANSLATORS: translated mail box name
'trash' => $l->t('Trash'),
// TRANSLATORS: translated mail box name
'junk' => $l->t('Junk'),
// TRANSLATORS: translated mail box name
'all' => $l->t('All'),
// TRANSLATORS: translated mail box name
'flagged' => $l->t('Favorites'),
Thomas Imbreckx
committed
$specialIds = $this->getSpecialFoldersIds(false);
foreach ($mailboxes as $i => $mailbox) {
if (in_array($mailbox->getFolderId(), $specialIds) === true) {
if (isset($map[$mailbox->getSpecialRole()])) {
$translatedDisplayName = $map[$mailbox->getSpecialRole()];
$mailboxes[$i]->setDisplayName((string)$translatedDisplayName);
}
/**
* Sort mailboxes
*
* Sort the array of mailboxes with
* - special use folders coming first in this order: all, inbox, flagged, drafts, sent, archive, junk, trash
* - 'normal' folders coming after that, sorted alphabetically
*/
protected function sortMailboxes() {
$mailboxes = $this->getMailboxes();
usort($mailboxes, function($a, $b) {
/**
* @var Mailbox $a
* @var Mailbox $b
*/
$roleA = $a->getSpecialRole();
$roleB = $b->getSpecialRole();
Thomas Imbreckx
committed
'all' => 0,
'inbox' => 1,
'flagged' => 2,
'drafts' => 3,
'sent' => 4,
'archive' => 5,
'junk' => 6,
'trash' => 7,
Thomas Imbreckx
committed
// if there is a flag unknown to us, we ignore it for sorting :
// the folder will be sorted by name like any other 'normal' folder
if (array_key_exists($roleA, $specialRolesOrder) === false) {
$roleA = null;
}
if (array_key_exists($roleB, $specialRolesOrder) === false) {
$roleB = null;
}
if ($roleA === null && $roleB !== null) {
return 1;
} elseif ($roleA !== null && $roleB === null) {
return -1;
} elseif ($roleA !== null && $roleB !== null) {
if ($roleA === $roleB) {
return strcasecmp($a->getdisplayName(), $b->getDisplayName());
} else {
return $specialRolesOrder[$roleA] - $specialRolesOrder[$roleB];
}
Thomas Imbreckx
committed
// we get here if $roleA === null && $roleB === null
return strcasecmp($a->getDisplayName(), $b->getDisplayName());
});
$this->mailboxes = $mailboxes;
/**
* Convert special security mode values into Horde parameters
*
protected function convertSslMode($sslMode) {
switch ($sslMode) {
case 'none':
return false;
}
* @return string|Horde_Mail_Rfc822_List
*/
public function getEmail() {
return $this->account->getEmail();
}
public function testConnectivity(Horde_Mail_Transport $transport) {
// connect to imap
$this->getImapConnection();
// connect to smtp
if ($transport instanceof Horde_Mail_Transport_Smtphorde) {
$transport->getSMTPObject();
/**
* Factory method for creating new messages
*
*/
public function newMessage() {
return new Message();
}
/**
* Factory method for creating new reply messages
*
* @return ReplyMessage
*/
public function newReplyMessage() {
return new ReplyMessage();
}
}