mirror of
https://github.com/chylex/Nextcloud-News.git
synced 2025-01-22 13:46:00 +01:00
b88e1a546a
Signed-off-by: Benjamin Brahmer <info@b-brahmer.de>
402 lines
12 KiB
PHP
402 lines
12 KiB
PHP
<?php
|
|
/**
|
|
* Nextcloud - News
|
|
*
|
|
* This file is licensed under the Affero General Public License version 3 or
|
|
* later. See the COPYING file.
|
|
*
|
|
* @author Alessandro Cosentino <cosenal@gmail.com>
|
|
* @author Bernhard Posselt <dev@bernhard-posselt.com>
|
|
* @copyright 2020 Sean Molenaar
|
|
*/
|
|
|
|
namespace OCA\News\Service;
|
|
|
|
use OCA\News\AppInfo\Application;
|
|
use OCA\News\Db\Feed;
|
|
use OCA\News\Db\ListType;
|
|
use OCA\News\Db\Item;
|
|
use OCA\News\Db\ItemMapperV2;
|
|
use OCA\News\Service\Exceptions\ServiceConflictException;
|
|
use OCA\News\Service\Exceptions\ServiceNotFoundException;
|
|
use OCA\News\Service\Exceptions\ServiceValidationException;
|
|
use OCP\AppFramework\Db\DoesNotExistException;
|
|
use OCP\AppFramework\Db\Entity;
|
|
use OCP\AppFramework\Db\MultipleObjectsReturnedException;
|
|
use OCP\IConfig;
|
|
use Psr\Log\LoggerInterface;
|
|
|
|
/**
|
|
* Class ItemService
|
|
*
|
|
* @package OCA\News\Service
|
|
*/
|
|
class ItemServiceV2 extends Service
|
|
{
|
|
|
|
/**
|
|
* @var IConfig
|
|
*/
|
|
protected $config;
|
|
|
|
/**
|
|
* ItemService constructor.
|
|
*
|
|
* @param ItemMapperV2 $mapper
|
|
* @param IConfig $config
|
|
* @param LoggerInterface $logger
|
|
*/
|
|
public function __construct(
|
|
ItemMapperV2 $mapper,
|
|
IConfig $config,
|
|
LoggerInterface $logger
|
|
) {
|
|
parent::__construct($mapper, $logger);
|
|
$this->config = $config;
|
|
}
|
|
|
|
/**
|
|
* Finds all items of a user
|
|
*
|
|
* @param string $userId The ID/name of the user
|
|
* @param array $params Filter parameters
|
|
*
|
|
*
|
|
* @return Item[]
|
|
*/
|
|
public function findAllForUser(string $userId, array $params = []): array
|
|
{
|
|
return $this->mapper->findAllFromUser($userId, $params);
|
|
}
|
|
|
|
/**
|
|
* Find all items
|
|
*
|
|
* @return Item[]
|
|
*/
|
|
public function findAll(): array
|
|
{
|
|
return $this->mapper->findAll();
|
|
}
|
|
|
|
/**
|
|
* Insert an item or update.
|
|
*
|
|
* @param Item $item
|
|
*
|
|
* @return Entity|Item The updated/inserted item
|
|
*/
|
|
public function insertOrUpdate(Item $item): Entity
|
|
{
|
|
try {
|
|
/** @var Item $db_item */
|
|
$db_item = $this->findByGuidHash($item->getFeedId(), $item->getGuidHash());
|
|
|
|
// Transfer user modifications
|
|
$item->setUnread($db_item->isUnread())
|
|
->setStarred($db_item->isStarred())
|
|
->setId($db_item->getId());
|
|
|
|
$item->generateSearchIndex();//generates fingerprint
|
|
|
|
// We don't want to update the database record if there is no
|
|
// change in the fetched item
|
|
if ($db_item->getFingerprint() === $item->getFingerprint()) {
|
|
$item->resetUpdatedFields();
|
|
}
|
|
|
|
return $this->mapper->update($item);
|
|
} catch (DoesNotExistException $exception) {
|
|
return $this->mapper->insert($item);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Return all starred items
|
|
*
|
|
* @param string $userId
|
|
*
|
|
* @return Item[]
|
|
*/
|
|
public function starred(string $userId): array
|
|
{
|
|
return $this->findAllForUser($userId, ['starred' => 1]);
|
|
}
|
|
|
|
/**
|
|
* Mark an item as read
|
|
*
|
|
* @param string $userId Item owner
|
|
* @param int $id Item ID
|
|
* @param bool $read
|
|
*
|
|
* @return Item
|
|
* @throws ServiceNotFoundException
|
|
* @throws ServiceConflictException
|
|
*/
|
|
public function read(string $userId, int $id, bool $read): Entity
|
|
{
|
|
/** @var Item $item */
|
|
$item = $this->find($userId, $id);
|
|
|
|
$item->setUnread(!$read);
|
|
|
|
return $this->mapper->update($item);
|
|
}
|
|
|
|
/**
|
|
* @param int|null $threshold
|
|
* @param bool|null $purgeUnread
|
|
*
|
|
* @return int|null Amount of deleted items or null if not applicable
|
|
*/
|
|
public function purgeOverThreshold(int $threshold = null, bool $purgeUnread = null): ?int
|
|
{
|
|
$threshold = (int) ($threshold ?? $this->config->getAppValue(
|
|
Application::NAME,
|
|
'autoPurgeCount',
|
|
Application::DEFAULT_SETTINGS['autoPurgeCount']
|
|
));
|
|
|
|
$purgeUnread = (bool) ($purgeUnread ?? $this->config->getAppValue(
|
|
Application::NAME,
|
|
'purgeUnread',
|
|
Application::DEFAULT_SETTINGS['purgeUnread']
|
|
));
|
|
|
|
if ($threshold <= 0) {
|
|
return null;
|
|
}
|
|
|
|
return $this->mapper->deleteOverThreshold($threshold, $purgeUnread);
|
|
}
|
|
/**
|
|
* Mark an item as starred
|
|
*
|
|
* @param string $userId Item owner
|
|
* @param int $id Item ID
|
|
* @param bool $starred
|
|
*
|
|
* @return Item
|
|
* @throws ServiceNotFoundException|ServiceConflictException
|
|
*/
|
|
public function star(string $userId, int $id, bool $starred): Entity
|
|
{
|
|
/** @var Item $item */
|
|
$item = $this->find($userId, $id);
|
|
|
|
$item->setStarred($starred);
|
|
|
|
return $this->mapper->update($item);
|
|
}
|
|
|
|
/**
|
|
* Mark an item as starred by GUID hash
|
|
*
|
|
* @param string $userId Item owner
|
|
* @param int $feedId Item ID
|
|
* @param string $guidHash
|
|
* @param bool $starred
|
|
*
|
|
* @return Item
|
|
* @throws ServiceConflictException
|
|
* @throws ServiceNotFoundException
|
|
*/
|
|
public function starByGuid(string $userId, int $feedId, string $guidHash, bool $starred): Entity
|
|
{
|
|
try {
|
|
$item = $this->mapper->findForUserByGuidHash($userId, $feedId, $guidHash);
|
|
} catch (DoesNotExistException $ex) {
|
|
throw ServiceNotFoundException::from($ex);
|
|
} catch (MultipleObjectsReturnedException $ex) {
|
|
throw ServiceConflictException::from($ex);
|
|
}
|
|
|
|
$item->setStarred($starred);
|
|
|
|
return $this->mapper->update($item);
|
|
}
|
|
|
|
/**
|
|
* Mark all items as read
|
|
*
|
|
* @param string $userId Item owner
|
|
* @param int $maxItemId
|
|
*
|
|
* @return int
|
|
*/
|
|
public function readAll(string $userId, int $maxItemId): int
|
|
{
|
|
return $this->mapper->readAll($userId, $maxItemId);
|
|
}
|
|
|
|
/**
|
|
* @param string $userId
|
|
*
|
|
* @return Item
|
|
*/
|
|
public function newest(string $userId): Entity
|
|
{
|
|
try {
|
|
return $this->mapper->newest($userId);
|
|
} catch (DoesNotExistException $e) {
|
|
throw ServiceNotFoundException::from($e);
|
|
} catch (MultipleObjectsReturnedException $e) {
|
|
throw ServiceConflictException::from($e);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @param int $feedId
|
|
* @param string $guidHash
|
|
*
|
|
* @return Item
|
|
*
|
|
* @throws DoesNotExistException
|
|
* @throws MultipleObjectsReturnedException
|
|
*/
|
|
public function findByGuidHash(int $feedId, string $guidHash): Entity
|
|
{
|
|
return $this->mapper->findByGuidHash($feedId, $guidHash);
|
|
}
|
|
|
|
/**
|
|
* Convenience method to find all items in a feed.
|
|
*
|
|
* @param string $userId
|
|
* @param int $feedId
|
|
*
|
|
* @return array
|
|
*/
|
|
public function findAllInFeed(string $userId, int $feedId): array
|
|
{
|
|
return $this->findAllInFeedAfter($userId, $feedId, PHP_INT_MIN, false);
|
|
}
|
|
|
|
/**
|
|
* Returns all new items in a feed
|
|
* @param string $userId the name of the user
|
|
* @param int $feedId the id of the feed
|
|
* @param int $updatedSince a timestamp with the minimal modification date
|
|
* @param boolean $hideRead if unread items should also be returned
|
|
*
|
|
* @return array of items
|
|
*/
|
|
public function findAllInFeedAfter(string $userId, int $feedId, int $updatedSince, bool $hideRead): array
|
|
{
|
|
return $this->mapper->findAllInFeedAfter($userId, $feedId, $updatedSince, $hideRead);
|
|
}
|
|
|
|
/**
|
|
* Returns all new items in a folder
|
|
* @param string $userId the name of the user
|
|
* @param int|null $folderId the id of the folder
|
|
* @param int $updatedSince a timestamp with the minimal modification date
|
|
* @param boolean $hideRead if unread items should also be returned
|
|
*
|
|
* @return array of items
|
|
*/
|
|
public function findAllInFolderAfter(string $userId, ?int $folderId, int $updatedSince, bool $hideRead): array
|
|
{
|
|
return $this->mapper->findAllInFolderAfter($userId, $folderId, $updatedSince, $hideRead);
|
|
}
|
|
|
|
/**
|
|
* Returns all new items of a type
|
|
*
|
|
* @param string $userId the name of the user
|
|
* @param int $feedType the type of feed items to fetch. (starred || unread)
|
|
* @param int $updatedSince a timestamp with the minimal modification date
|
|
*
|
|
* @return array of items
|
|
*
|
|
* @throws ServiceValidationException
|
|
*/
|
|
public function findAllAfter(string $userId, int $feedType, int $updatedSince): array
|
|
{
|
|
if (!in_array($feedType, [ListType::STARRED, ListType::UNREAD, ListType::ALL_ITEMS], true)) {
|
|
throw new ServiceValidationException('Trying to find in unknown type');
|
|
}
|
|
|
|
return $this->mapper->findAllAfter($userId, $feedType, $updatedSince);
|
|
}
|
|
|
|
|
|
/**
|
|
* Returns all items
|
|
*
|
|
* @param int $feedId the id of the feed
|
|
* @param int $limit how many items should be returned
|
|
* @param int $offset the offset
|
|
* @param boolean $hideRead if unread items should also be returned
|
|
* @param boolean $oldestFirst if it should be ordered by oldest first
|
|
* @param string $userId the name of the user
|
|
* @param string[] $search an array of keywords that the result should
|
|
* contain in either the author, title, link
|
|
* or body
|
|
*
|
|
* @return array of items
|
|
*/
|
|
public function findAllInFeedWithFilters(
|
|
string $userId,
|
|
int $feedId,
|
|
int $limit,
|
|
int $offset,
|
|
bool $hideRead,
|
|
bool $oldestFirst,
|
|
array $search = []
|
|
): array {
|
|
return $this->mapper->findAllFeed($userId, $feedId, $limit, $offset, $hideRead, $oldestFirst, $search);
|
|
}
|
|
/**
|
|
* Returns all items
|
|
*
|
|
* @param int|null $folderId the id of the folder
|
|
* @param int $limit how many items should be returned
|
|
* @param int $offset the offset
|
|
* @param boolean $hideRead if unread items should also be returned
|
|
* @param boolean $oldestFirst if it should be ordered by oldest first
|
|
* @param string $userId the name of the user
|
|
* @param string[] $search an array of keywords that the result should
|
|
* contain in either the author, title, link
|
|
* or body
|
|
*
|
|
* @return array of items
|
|
*/
|
|
public function findAllInFolderWithFilters(
|
|
string $userId,
|
|
?int $folderId,
|
|
int $limit,
|
|
int $offset,
|
|
bool $hideRead,
|
|
bool $oldestFirst,
|
|
array $search = []
|
|
): array {
|
|
return $this->mapper->findAllFolder($userId, $folderId, $limit, $offset, $hideRead, $oldestFirst, $search);
|
|
}
|
|
/**
|
|
* Returns all items
|
|
*
|
|
* @param int $type the type of the feed
|
|
* @param int $limit how many items should be returned
|
|
* @param int $offset the offset
|
|
* @param boolean $oldestFirst if it should be ordered by oldest first
|
|
* @param string $userId the name of the user
|
|
* @param string[] $search an array of keywords that the result should
|
|
* contain in either the author, title, link
|
|
* or body
|
|
*
|
|
* @return array of items
|
|
*/
|
|
public function findAllWithFilters(
|
|
string $userId,
|
|
int $type,
|
|
int $limit,
|
|
int $offset,
|
|
bool $oldestFirst,
|
|
array $search = []
|
|
): array {
|
|
return $this->mapper->findAllItems($userId, $type, $limit, $offset, $oldestFirst, $search);
|
|
}
|
|
}
|