diff --git a/appinfo/info.xml b/appinfo/info.xml index 29a3ba0a..76a3148f 100644 --- a/appinfo/info.xml +++ b/appinfo/info.xml @@ -46,12 +46,15 @@ + OCA\Social\Command\AccountCreate + OCA\Social\Command\AccountFollowing OCA\Social\Command\CacheRefresh OCA\Social\Command\CheckInstall OCA\Social\Command\Fediverse OCA\Social\Command\NoteCreate OCA\Social\Command\NoteBoost OCA\Social\Command\Reset + OCA\Social\Command\Timeline OCA\Social\Command\QueueStatus OCA\Social\Command\QueueProcess diff --git a/lib/AP.php b/lib/AP.php index 54c7382d..418c0578 100644 --- a/lib/AP.php +++ b/lib/AP.php @@ -219,6 +219,7 @@ class AP { } $this->getObjectFromData($data, $item, $level); + $this->getActorFromData($data, $item, $level); return $item; } @@ -250,6 +251,27 @@ class AP { } + /** + * @param array $data + * @param ACore $item + * @param int $level + * + * @throws RedundancyLimitException + * @throws SocialAppConfigException + */ + public function getActorFromData(array $data, ACore &$item, int $level) { + try { + $actorData = $this->getArray('actor_info', $data, []); + if (!empty($actorData)) { + /** @var Person $actor */ + $actor = $this->getItemFromData($actorData, $item, $level); + $item->setActor($actor); + } + } catch (ItemUnknownException $e) { + } + } + + /** * @param array $data * diff --git a/lib/Command/AccountCreate.php b/lib/Command/AccountCreate.php new file mode 100644 index 00000000..c42d4757 --- /dev/null +++ b/lib/Command/AccountCreate.php @@ -0,0 +1,123 @@ + + * @copyright 2018, Maxence Lange + * @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 . + * + */ + + +namespace OCA\Social\Command; + + +use Exception; +use OC\Core\Command\Base; +use OCA\Social\Service\AccountService; +use OCA\Social\Service\CacheActorService; +use OCA\Social\Service\ConfigService; +use OCA\Social\Service\MiscService; +use OCP\IUserManager; +use Symfony\Component\Console\Input\InputArgument; +use Symfony\Component\Console\Input\InputInterface; +use Symfony\Component\Console\Input\InputOption; +use Symfony\Component\Console\Output\OutputInterface; + + +class AccountCreate extends Base { + + + /** @var IUserManager */ + private $userManager; + + /** @var AccountService */ + private $accountService; + + /** @var CacheActorService */ + private $cacheActorService; + + /** @var ConfigService */ + private $configService; + + /** @var MiscService */ + private $miscService; + + + /** + * CacheUpdate constructor. + * + * @param IUserManager $userManager + * @param AccountService $accountService + * @param CacheActorService $cacheActorService + * @param ConfigService $configService + * @param MiscService $miscService + */ + public function __construct( + IUserManager $userManager, AccountService $accountService, + CacheActorService $cacheActorService, ConfigService $configService, MiscService $miscService + ) { + parent::__construct(); + + $this->userManager = $userManager; + + $this->accountService = $accountService; + $this->cacheActorService = $cacheActorService; + $this->configService = $configService; + $this->miscService = $miscService; + } + + + /** + * + */ + protected function configure() { + parent::configure(); + $this->setName('social:account:create') + ->addArgument('userId', InputArgument::REQUIRED, 'Nextcloud username of the account') + ->addOption('handle', '', InputOption::VALUE_REQUIRED, 'social handle') + ->setDescription('Create a new social account'); + } + + + /** + * @param InputInterface $input + * @param OutputInterface $output + * + * @throws Exception + */ + protected function execute(InputInterface $input, OutputInterface $output) { + $userId = $input->getArgument('userId'); + + if (($handle = $input->getOption('handle')) === null) { + $handle = $userId; + } + + if ($this->userManager->get($userId) === null) { + throw new Exception('Unknown user'); + } + + $this->accountService->createActor($userId, $handle); + } + +} + diff --git a/lib/Command/AccountFollowing.php b/lib/Command/AccountFollowing.php new file mode 100644 index 00000000..5833ef11 --- /dev/null +++ b/lib/Command/AccountFollowing.php @@ -0,0 +1,127 @@ + + * @copyright 2018, Maxence Lange + * @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 . + * + */ + + +namespace OCA\Social\Command; + + +use Exception; +use OC\Core\Command\Base; +use OCA\Social\Service\AccountService; +use OCA\Social\Service\CacheActorService; +use OCA\Social\Service\ConfigService; +use OCA\Social\Service\FollowService; +use OCA\Social\Service\MiscService; +use Symfony\Component\Console\Input\InputArgument; +use Symfony\Component\Console\Input\InputInterface; +use Symfony\Component\Console\Input\InputOption; +use Symfony\Component\Console\Output\OutputInterface; + + +class AccountFollowing extends Base { + + + /** @var AccountService */ + private $accountService; + + /** @var CacheActorService */ + private $cacheActorService; + + /** @var FollowService */ + private $followService; + + /** @var ConfigService */ + private $configService; + + /** @var MiscService */ + private $miscService; + + + /** + * CacheUpdate constructor. + * + * @param AccountService $accountService + * @param CacheActorService $cacheActorService + * @param FollowService $followService + * @param ConfigService $configService + * @param MiscService $miscService + */ + public function __construct( + AccountService $accountService, CacheActorService $cacheActorService, + FollowService $followService, ConfigService $configService, MiscService $miscService + ) { + parent::__construct(); + + $this->accountService = $accountService; + $this->cacheActorService = $cacheActorService; + $this->followService = $followService; + $this->configService = $configService; + $this->miscService = $miscService; + } + + + /** + * + */ + protected function configure() { + parent::configure(); + $this->setName('social:account:following') + ->addArgument('userId', InputArgument::REQUIRED, 'Nextcloud userid') + ->addArgument('account', InputArgument::REQUIRED, 'Account to follow') + ->addOption('local', '', InputOption::VALUE_NONE, 'account is local') + ->addOption('unfollow', '', InputOption::VALUE_NONE, 'unfollow') + ->setDescription('Following a new account'); + } + + + /** + * @param InputInterface $input + * @param OutputInterface $output + * + * @throws Exception + */ + protected function execute(InputInterface $input, OutputInterface $output) { + $userId = $input->getArgument('userId'); + $account = $input->getArgument('account'); + + $actor = $this->accountService->getActor($userId); + if ($input->getOption('local')) { + $local = $this->cacheActorService->getFromLocalAccount($account); + $account = $local->getAccount(); + } + + if ($input->getOption('unfollow')) { + $this->followService->unfollowAccount($actor, $account); + } else { + $this->followService->followAccount($actor, $account); + } + } + +} + diff --git a/lib/Command/Timeline.php b/lib/Command/Timeline.php new file mode 100644 index 00000000..d2120ead --- /dev/null +++ b/lib/Command/Timeline.php @@ -0,0 +1,267 @@ + + * @copyright 2018, Maxence Lange + * @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 . + * + */ + + +namespace OCA\Social\Command; + + +use daita\MySmallPhpTools\Exceptions\CacheItemNotFoundException; +use Exception; +use OC\Core\Command\Base; +use OCA\Social\AP; +use OCA\Social\Db\StreamRequest; +use OCA\Social\Exceptions\ItemUnknownException; +use OCA\Social\Exceptions\RedundancyLimitException; +use OCA\Social\Exceptions\SocialAppConfigException; +use OCA\Social\Model\ActivityPub\Actor\Person; +use OCA\Social\Model\ActivityPub\Stream; +use OCA\Social\Service\AccountService; +use OCA\Social\Service\CacheActorService; +use OCA\Social\Service\ConfigService; +use OCA\Social\Service\MiscService; +use OCP\IUserManager; +use Symfony\Component\Console\Helper\Table; +use Symfony\Component\Console\Input\InputArgument; +use Symfony\Component\Console\Input\InputInterface; +use Symfony\Component\Console\Input\InputOption; +use Symfony\Component\Console\Output\ConsoleOutput; +use Symfony\Component\Console\Output\OutputInterface; + + +/** + * Class Stream + * + * @package OCA\Social\Command + */ +class Timeline extends Base { + + /** @var IUserManager */ + private $userManager; + + /** @var StreamRequest */ + private $streamRequest; + + /** @var AccountService */ + private $accountService; + + /** @var ConfigService */ + private $configService; + + /** @var MiscService */ + private $miscService; + + + /** @var OutputInterface */ + private $output; + + /** @var bool */ + private $asJson; + + /** @var int */ + private $count; + + + /** + * Stream constructor. + * + * @param IUserManager $userManager + * @param StreamRequest $streamRequest + * @param AccountService $accountService + * @param ConfigService $configService + * @param MiscService $miscService + */ + public function __construct( + IUserManager $userManager, StreamRequest $streamRequest, + AccountService $accountService, ConfigService $configService, + MiscService $miscService + ) { + parent::__construct(); + + $this->userManager = $userManager; + $this->streamRequest = $streamRequest; + $this->accountService = $accountService; + $this->configService = $configService; + $this->miscService = $miscService; + } + + + /** + * + */ + protected function configure() { + parent::configure(); + $this->setName('social:stream') + ->addArgument('userId', InputArgument::REQUIRED, 'viewer') + ->addArgument('timeline', InputArgument::REQUIRED, 'timeline') + ->addOption('count', '', InputOption::VALUE_REQUIRED, 'number of elements', '5') + ->addOption('json', '', InputOption::VALUE_NONE, 'return JSON format') + ->setDescription('Get stream by timeline and viewer'); + } + + + /** + * @param InputInterface $input + * @param OutputInterface $output + * + * @throws Exception + */ + protected function execute(InputInterface $input, OutputInterface $output) { + $output = new ConsoleOutput(); + $this->output = $output->section(); + + $this->asJson = $input->getOption('json'); + $this->count = intval($input->getOption('count')); + + $timeline = $input->getArgument('timeline'); + + $userId = $input->getArgument('userId'); + if ($this->userManager->get($userId) === null) { + throw new Exception('Unknown user'); + } + + $actor = $this->accountService->getActor($userId); + + $this->outputActor($actor); + $this->displayStream($actor, $timeline); + } + + + /** + * @param Person $actor + */ + private function outputActor(Person $actor) { + if ($this->asJson) { + return; + } + + $this->output->writeln('Account: ' . $actor->getAccount()); + $this->output->writeln('Id: ' . $actor->getId()); + $this->output->writeln(''); + + } + + + /** + * @param Person $actor + * @param string $timeline + * + * @throws Exception + */ + private function displayStream(Person $actor, string $timeline) { + switch ($timeline) { + case 'home': + $stream = $this->streamRequest->getTimelineHome($actor, 0, $this->count); + $this->outputStream($stream); + break; + + case 'direct': + $stream = $this->streamRequest->getTimelineDirect($actor, 0, $this->count); + $this->outputStream($stream); + break; + + case 'notifications': + $stream = $this->streamRequest->getTimelineNotifications($actor, 0, $this->count); + $this->outputStream($stream); + break; + + case 'local': + $stream = $this->streamRequest->getTimelineGlobal(0, $this->count, true); + $this->outputStream($stream); + break; + + case 'global': + $stream = $this->streamRequest->getTimelineGlobal(0, $this->count, false); + $this->outputStream($stream); + break; + + default: + throw new Exception( + 'Unknown timeline. Try home, direct, local, global, notification.' + ); + } + } + + + /** + * @param Stream[] $stream + */ + private function outputStream(array $stream) { + if ($this->asJson) { + $this->output->writeln(json_encode($stream, JSON_PRETTY_PRINT)); + } + + $table = new Table($this->output); + $table->setHeaders(['Id', 'Source', 'Type', 'Author', 'Content']); + $table->render(); + $this->output->writeln(''); + + foreach ($stream as $item) { + $objectId = $item->getObjectId(); + $cache = $item->getCache(); + $content = ''; + $author = ''; + if ($objectId !== '' && $cache->hasItem($objectId)) { + try { + $cachedObject = $cache->getItem($objectId) + ->getObject(); + + /** @var Stream $cachedItem */ + $cachedItem = AP::$activityPub->getItemFromData($cachedObject); + $content = $cachedItem->getContent(); + $author = $cachedItem->getActor() + ->getAccount(); + } catch (CacheItemNotFoundException $e) { + } catch (ItemUnknownException $e) { + } catch (RedundancyLimitException $e) { + } catch (SocialAppConfigException $e) { + } + } else { + $content = $item->getContent(); + $author = $item->getActor() + ->getAccount(); + } + + $table->appendRow( + [ + '' . $item->getId() . '', + '' . $item->getActor() + ->getAccount() . '', + $item->getType(), + '' . $author . '', + $content, + ] + ); + } + + + } + + +} + diff --git a/lib/Service/FollowService.php b/lib/Service/FollowService.php index ec4b5b67..512dbd83 100644 --- a/lib/Service/FollowService.php +++ b/lib/Service/FollowService.php @@ -48,6 +48,7 @@ use OCA\Social\Exceptions\RequestResultSizeException; use OCA\Social\Exceptions\RequestServerException; use OCA\Social\Exceptions\SocialAppConfigException; use OCA\Social\Exceptions\ItemUnknownException; +use OCA\Social\Exceptions\UnauthorizedFediverseException; use OCA\Social\Exceptions\UrlCloudException; use OCA\Social\Model\ActivityPub\Object\Follow; use OCA\Social\Model\ActivityPub\Activity\Undo; @@ -131,6 +132,7 @@ class FollowService { * @throws RequestResultSizeException * @throws RequestServerException * @throws RequestResultNotJsonException + * @throws UnauthorizedFediverseException */ public function followAccount(Person $actor, string $account) { $remoteActor = $this->cacheActorService->getFromAccount($account);