mail/lib/Db/LocalMessageMapper.php

279 строки
8.7 KiB
PHP

<?php
declare(strict_types=1);
/**
* @copyright 2022 Anna Larch <anna@nextcloud.com>
*
* @author 2022 Anna Larch <anna@nextcloud.com>
*
* @license GNU AGPL version 3 or any later version
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU Affero 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 Affero General Public License for more details.
*
* You should have received a copy of the GNU Affero General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
namespace OCA\Mail\Db;
use OCP\AppFramework\Db\DoesNotExistException;
use OCP\DB\Exception as DBException;
use Throwable;
use function array_map;
use OCP\AppFramework\Db\QBMapper;
use OCP\DB\QueryBuilder\IQueryBuilder;
use OCP\IDBConnection;
use function array_merge;
/**
* @template-extends QBMapper<LocalMessage>
*/
class LocalMessageMapper extends QBMapper {
/** @var LocalAttachmentMapper */
private $attachmentMapper;
/** @var RecipientMapper */
private $recipientMapper;
public function __construct(IDBConnection $db,
LocalAttachmentMapper $attachmentMapper,
RecipientMapper $recipientMapper) {
parent::__construct($db, 'mail_local_messages');
$this->recipientMapper = $recipientMapper;
$this->attachmentMapper = $attachmentMapper;
}
/**
* @param string $userId
* @return LocalMessage[]
* @throws DBException
*/
public function getAllForUser(string $userId, int $type = LocalMessage::TYPE_OUTGOING): array {
$qb = $this->db->getQueryBuilder();
$qb->select('m.*')
->from('mail_accounts', 'a')
->join('a', $this->getTableName(), 'm', $qb->expr()->eq('m.account_id', 'a.id'))
->where(
$qb->expr()->eq('a.user_id', $qb->createNamedParameter($userId, IQueryBuilder::PARAM_STR), IQueryBuilder::PARAM_STR),
$qb->expr()->eq('m.type', $qb->createNamedParameter($type, IQueryBuilder::PARAM_INT), IQueryBuilder::PARAM_INT)
);
$rows = $qb->execute();
$results = [];
$ids = [];
while (($row = $rows->fetch()) !== false) {
$results[] = $this->mapRowToEntity($row);
$ids[] = $row['id'];
}
$rows->closeCursor();
if (empty($ids)) {
return [];
}
$attachments = $this->attachmentMapper->findByLocalMessageIds($ids);
$recipients = $this->recipientMapper->findByLocalMessageIds($ids);
$recipientMap = [];
foreach ($recipients as $r) {
$recipientMap[$r->getLocalMessageId()][] = $r;
}
$attachmentMap = [];
foreach ($attachments as $a) {
$attachmentMap[$a->getLocalMessageId()][] = $a;
}
return array_map(static function ($localMessage) use ($attachmentMap, $recipientMap) {
$localMessage->setAttachments($attachmentMap[$localMessage->getId()] ?? []);
$localMessage->setRecipients($recipientMap[$localMessage->getId()] ?? []);
return $localMessage;
}, $results);
}
/**
* @throws DoesNotExistException
*/
public function findById(int $id, string $userId): LocalMessage {
$qb = $this->db->getQueryBuilder();
$qb->select('m.*')
->from('mail_accounts', 'a')
->join('a', $this->getTableName(), 'm', $qb->expr()->eq('m.account_id', 'a.id'))
->where(
$qb->expr()->eq('a.user_id', $qb->createNamedParameter($userId, IQueryBuilder::PARAM_STR), IQueryBuilder::PARAM_STR),
$qb->expr()->eq('m.id', $qb->createNamedParameter($id, IQueryBuilder::PARAM_INT), IQueryBuilder::PARAM_INT)
);
$entity = $this->findEntity($qb);
$entity->setAttachments($this->attachmentMapper->findByLocalMessageId($userId, $id));
$entity->setRecipients($this->recipientMapper->findByLocalMessageId($id));
return $entity;
}
/**
* Find all messages that should be sent
*
* @param int $time upper bound send time stamp
*
* @return LocalMessage[]
*/
public function findDue(int $time, int $type = LocalMessage::TYPE_OUTGOING): array {
$qb = $this->db->getQueryBuilder();
$select = $qb->select('*')
->from($this->getTableName())
->where(
$qb->expr()->isNotNull('send_at'),
$qb->expr()->eq('type', $qb->createNamedParameter($type, IQueryBuilder::PARAM_INT), IQueryBuilder::PARAM_INT),
$qb->expr()->lte('send_at', $qb->createNamedParameter($time, IQueryBuilder::PARAM_INT), IQueryBuilder::PARAM_INT),
$qb->expr()->orX(
$qb->expr()->isNull('failed'),
$qb->expr()->eq('failed', $qb->createNamedParameter(false, IQueryBuilder::PARAM_BOOL), IQueryBuilder::PARAM_BOOL),
)
)
->orderBy('send_at', 'asc');
$messages = $this->findEntities($select);
if (empty($messages)) {
return [];
}
$ids = array_map(static function (LocalMessage $message) {
return $message->getId();
}, $messages);
$attachments = $this->attachmentMapper->findByLocalMessageIds($ids);
$recipients = $this->recipientMapper->findByLocalMessageIds($ids);
$recipientMap = [];
foreach ($recipients as $r) {
$recipientMap[$r->getLocalMessageId()][] = $r;
}
$attachmentMap = [];
foreach ($attachments as $a) {
$attachmentMap[$a->getLocalMessageId()][] = $a;
}
return array_map(static function ($localMessage) use ($attachmentMap, $recipientMap) {
$localMessage->setAttachments($attachmentMap[$localMessage->getId()] ?? []);
$localMessage->setRecipients($recipientMap[$localMessage->getId()] ?? []);
return $localMessage;
}, $messages);
}
/**
* Find all messages that should be sent
*
* @param int $time upper bound send time stamp
*
* @return LocalMessage[]
*/
public function findDueDrafts(int $time): array {
$qb = $this->db->getQueryBuilder();
$select = $qb->select('*')
->from($this->getTableName())
->where(
$qb->expr()->isNotNull('send_at'),
$qb->expr()->eq('type', $qb->createNamedParameter(LocalMessage::TYPE_DRAFT, IQueryBuilder::PARAM_INT), IQueryBuilder::PARAM_INT),
$qb->expr()->lte('updated_at', $qb->createNamedParameter($time - 300, IQueryBuilder::PARAM_INT), IQueryBuilder::PARAM_INT),
$qb->expr()->orX(
$qb->expr()->isNull('failed'),
$qb->expr()->eq('failed', $qb->createNamedParameter(false, IQueryBuilder::PARAM_BOOL), IQueryBuilder::PARAM_BOOL),
)
)
->orderBy('updated_at', 'asc')
->orderBy('account_id', 'asc');
$messages = $this->findEntities($select);
if (empty($messages)) {
return [];
}
$ids = array_map(static function (LocalMessage $message) {
return $message->getId();
}, $messages);
$attachments = $this->attachmentMapper->findByLocalMessageIds($ids);
$recipients = $this->recipientMapper->findByLocalMessageIds($ids);
$recipientMap = [];
foreach ($recipients as $r) {
$recipientMap[$r->getLocalMessageId()][] = $r;
}
$attachmentMap = [];
foreach ($attachments as $a) {
$attachmentMap[$a->getLocalMessageId()][] = $a;
}
return array_map(static function ($localMessage) use ($attachmentMap, $recipientMap) {
$localMessage->setAttachments($attachmentMap[$localMessage->getId()] ?? []);
$localMessage->setRecipients($recipientMap[$localMessage->getId()] ?? []);
return $localMessage;
}, $messages);
}
/**
* @param Recipient[] $to
* @param Recipient[] $cc
* @param Recipient[] $bcc
*/
public function saveWithRecipients(LocalMessage $message, array $to, array $cc, array $bcc): LocalMessage {
$this->db->beginTransaction();
try {
$message = $this->insert($message);
$this->recipientMapper->saveRecipients($message->getId(), $to);
$this->recipientMapper->saveRecipients($message->getId(), $cc);
$this->recipientMapper->saveRecipients($message->getId(), $bcc);
$this->db->commit();
} catch (Throwable $e) {
$this->db->rollBack();
throw $e;
}
$message->setRecipients(array_merge(
$to,
$cc,
$bcc,
));
return $message;
}
/**
* @param Recipient[] $to
* @param Recipient[] $cc
* @param Recipient[] $bcc
*/
public function updateWithRecipients(LocalMessage $message, array $to, array $cc, array $bcc): LocalMessage {
$this->db->beginTransaction();
try {
$message = $this->update($message);
$this->recipientMapper->updateRecipients($message->getId(), $message->getRecipients(), $to, $cc, $bcc);
$this->db->commit();
} catch (Throwable $e) {
$this->db->rollBack();
throw $e;
}
$recipients = $this->recipientMapper->findByLocalMessageId($message->getId());
$message->setRecipients($recipients);
return $message;
}
public function deleteWithRecipients(LocalMessage $message): void {
$this->db->beginTransaction();
try {
$this->recipientMapper->deleteForLocalMessage($message->getId());
$this->delete($message);
$this->db->commit();
} catch (Throwable $e) {
$this->db->rollBack();
throw $e;
}
}
}