2020-03-12 17:59:13 +00:00
|
|
|
<?php
|
|
|
|
|
2021-10-10 09:26:18 +01:00
|
|
|
declare(strict_types = 1);
|
|
|
|
|
2020-05-20 17:53:53 +01:00
|
|
|
// {{{ License
|
2020-10-10 19:25:49 +01:00
|
|
|
|
2020-05-20 17:53:53 +01:00
|
|
|
// This file is part of GNU social - https://www.gnu.org/software/social
|
|
|
|
//
|
|
|
|
// GNU social 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.
|
|
|
|
//
|
|
|
|
// GNU social 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 GNU social. If not, see <http://www.gnu.org/licenses/>.
|
2020-10-10 19:25:49 +01:00
|
|
|
|
2020-05-20 17:53:53 +01:00
|
|
|
// }}}
|
|
|
|
|
2020-03-21 20:18:05 +00:00
|
|
|
/**
|
|
|
|
* Handle network public feed
|
|
|
|
*
|
|
|
|
* @package GNUsocial
|
|
|
|
* @category Controller
|
|
|
|
*
|
2021-02-19 23:29:43 +00:00
|
|
|
* @author Hugo Sales <hugo@hsal.es>
|
2020-09-04 00:15:18 +01:00
|
|
|
* @author Eliseu Amaro <eliseu@fc.up.pt>
|
2021-02-19 23:29:43 +00:00
|
|
|
* @copyright 2020-2021 Free Software Foundation, Inc http://www.fsf.org
|
2020-03-21 20:18:05 +00:00
|
|
|
* @license https://www.gnu.org/licenses/agpl.html GNU AGPL v3 or later
|
|
|
|
*/
|
|
|
|
|
2020-03-12 17:59:13 +00:00
|
|
|
namespace App\Controller;
|
|
|
|
|
2021-11-26 23:31:53 +00:00
|
|
|
use App\Core\Cache;
|
2020-07-06 21:51:08 +01:00
|
|
|
use App\Core\Controller;
|
2020-08-15 06:47:45 +01:00
|
|
|
use App\Core\DB\DB;
|
2021-02-18 22:29:55 +00:00
|
|
|
use App\Core\Event;
|
2021-11-26 23:31:53 +00:00
|
|
|
use App\Core\Form;
|
2021-04-14 20:54:38 +01:00
|
|
|
use function App\Core\I18n\_m;
|
2021-11-26 23:31:53 +00:00
|
|
|
use App\Core\Router\Router;
|
2021-04-15 23:26:05 +01:00
|
|
|
use App\Core\VisibilityScope;
|
2021-11-26 23:31:53 +00:00
|
|
|
use App\Entity\Feed;
|
2020-12-02 22:57:32 +00:00
|
|
|
use App\Entity\Note;
|
2020-09-04 00:15:18 +01:00
|
|
|
use App\Util\Common;
|
2020-09-05 03:37:05 +01:00
|
|
|
use App\Util\Exception\ClientException;
|
2021-05-23 20:56:45 +01:00
|
|
|
use App\Util\Exception\NotFoundException;
|
2021-11-25 23:08:30 +00:00
|
|
|
use App\Util\Exception\NotImplementedException;
|
2021-11-26 23:31:53 +00:00
|
|
|
use App\Util\Exception\RedirectException;
|
|
|
|
use Functional as F;
|
|
|
|
use Symfony\Component\Form\Extension\Core\Type\IntegerType;
|
|
|
|
use Symfony\Component\Form\Extension\Core\Type\SubmitType;
|
|
|
|
use Symfony\Component\Form\Extension\Core\Type\TextType;
|
2020-08-15 06:47:45 +01:00
|
|
|
use Symfony\Component\HttpFoundation\Request;
|
2021-11-26 23:31:53 +00:00
|
|
|
use Symfony\Component\Routing\Exception\ResourceNotFoundException;
|
2020-03-15 21:21:11 +00:00
|
|
|
|
2021-11-26 20:37:17 +00:00
|
|
|
class Feeds extends Controller
|
2020-03-12 17:59:13 +00:00
|
|
|
{
|
2021-04-29 17:42:06 +01:00
|
|
|
// Can't have constants inside herestring
|
2021-11-15 17:06:34 +00:00
|
|
|
private $public_scope = VisibilityScope::PUBLIC;
|
|
|
|
private $instance_scope = VisibilityScope::PUBLIC | VisibilityScope::SITE;
|
|
|
|
private $message_scope = VisibilityScope::MESSAGE;
|
2021-11-08 13:44:35 +00:00
|
|
|
private $subscriber_scope = VisibilityScope::PUBLIC | VisibilityScope::SUBSCRIBER;
|
2020-09-05 03:37:05 +01:00
|
|
|
|
2020-08-27 03:28:34 +01:00
|
|
|
public function public(Request $request)
|
|
|
|
{
|
2021-04-14 20:54:38 +01:00
|
|
|
$notes = Note::getAllNotes($this->instance_scope);
|
2021-02-18 22:29:55 +00:00
|
|
|
|
2021-09-06 19:49:03 +01:00
|
|
|
$notes_out = null;
|
2021-08-07 22:52:00 +01:00
|
|
|
Event::handle('FormatNoteList', [$notes, &$notes_out]);
|
2021-02-18 22:29:55 +00:00
|
|
|
|
2020-08-27 03:28:34 +01:00
|
|
|
return [
|
2021-11-26 20:37:17 +00:00
|
|
|
'_template' => 'feeds/feed.html.twig',
|
2021-10-10 09:26:18 +01:00
|
|
|
'notes' => $notes_out,
|
2021-11-26 13:05:23 +00:00
|
|
|
'page_title' => 'Public feed',
|
2020-08-27 03:28:34 +01:00
|
|
|
];
|
|
|
|
}
|
|
|
|
|
2020-09-05 03:37:05 +01:00
|
|
|
public function home(Request $request, string $nickname)
|
2020-07-06 21:51:08 +01:00
|
|
|
{
|
2021-05-23 20:56:45 +01:00
|
|
|
try {
|
2021-09-18 03:22:27 +01:00
|
|
|
$target = DB::findOneBy('actor', ['nickname' => $nickname]);
|
2021-05-23 20:56:45 +01:00
|
|
|
} catch (NotFoundException) {
|
2020-09-05 03:37:05 +01:00
|
|
|
throw new ClientException(_m('User {nickname} doesn\'t exist', ['{nickname}' => $nickname]));
|
|
|
|
}
|
|
|
|
|
2021-11-15 17:06:34 +00:00
|
|
|
// TODO Handle replies in home stream
|
2020-09-05 03:37:05 +01:00
|
|
|
$query = <<<END
|
2021-10-10 09:26:18 +01:00
|
|
|
-- Select notes from:
|
|
|
|
select note.* from note left join -- left join ensures all returned notes' ids are not null
|
|
|
|
(
|
2021-11-08 13:44:35 +00:00
|
|
|
-- Subscribed by target
|
|
|
|
select n.id from note n inner join subscription f on n.actor_id = f.subscribed
|
|
|
|
where f.subscriber = :target_actor_id
|
2021-10-10 09:26:18 +01:00
|
|
|
union all
|
|
|
|
-- Replies to notes by target
|
2021-11-15 17:06:34 +00:00
|
|
|
-- select n.id from note n inner join note nr on nr.id = nr.reply_to
|
|
|
|
-- union all
|
2021-10-10 09:26:18 +01:00
|
|
|
-- Notifications to target
|
|
|
|
select a.activity_id from notification a inner join note n on a.activity_id = n.id
|
|
|
|
union all
|
2021-11-08 13:44:35 +00:00
|
|
|
-- Notes in groups target subscriptions
|
2021-10-10 09:26:18 +01:00
|
|
|
select gi.activity_id from group_inbox gi inner join group_member gm on gi.group_id = gm.group_id
|
|
|
|
where gm.actor_id = :target_actor_id
|
|
|
|
)
|
|
|
|
as s on s.id = note.id
|
|
|
|
where
|
|
|
|
-- Remove direct messages
|
|
|
|
note.scope <> {$this->message_scope}
|
|
|
|
order by note.modified DESC
|
|
|
|
END;
|
2021-11-08 20:21:41 +00:00
|
|
|
$notes = DB::sql($query, ['target_actor_id' => $target->getId()]);
|
2020-09-05 03:37:05 +01:00
|
|
|
|
2021-09-06 19:49:03 +01:00
|
|
|
$notes_out = null;
|
2021-08-07 22:52:00 +01:00
|
|
|
Event::handle('FormatNoteList', [$notes, &$notes_out]);
|
2021-04-14 20:54:38 +01:00
|
|
|
|
2020-07-06 23:16:50 +01:00
|
|
|
return [
|
2021-11-26 20:37:17 +00:00
|
|
|
'_template' => 'feeds/feed.html.twig',
|
2021-10-10 09:26:18 +01:00
|
|
|
'notes' => $notes_out,
|
2021-11-26 13:05:23 +00:00
|
|
|
'page_title' => 'Home feed',
|
2020-09-04 00:15:18 +01:00
|
|
|
];
|
|
|
|
}
|
|
|
|
|
|
|
|
public function network(Request $request)
|
|
|
|
{
|
2021-05-23 20:56:45 +01:00
|
|
|
$notes = Note::getAllNotes($this->public_scope);
|
2020-09-04 00:15:18 +01:00
|
|
|
|
2021-09-06 19:49:03 +01:00
|
|
|
$notes_out = null;
|
2021-08-07 22:52:00 +01:00
|
|
|
Event::handle('FormatNoteList', [$notes, &$notes_out]);
|
2020-09-04 00:15:18 +01:00
|
|
|
|
|
|
|
return [
|
2021-11-26 20:37:17 +00:00
|
|
|
'_template' => 'feeds/feed.html.twig',
|
2021-10-10 09:26:18 +01:00
|
|
|
'notes' => $notes_out,
|
2021-11-26 13:05:23 +00:00
|
|
|
'page_title' => 'Network feed',
|
2020-07-06 23:16:50 +01:00
|
|
|
];
|
2020-03-12 17:59:13 +00:00
|
|
|
}
|
2020-09-05 05:10:33 +01:00
|
|
|
|
2021-11-26 23:31:53 +00:00
|
|
|
public function edit_feeds(Request $request)
|
|
|
|
{
|
|
|
|
$user = Common::ensureLoggedIn();
|
|
|
|
$key = Feed::cacheKey($user);
|
|
|
|
$feeds = Feed::getFeeds($user);
|
|
|
|
|
|
|
|
$form_definitions = [];
|
|
|
|
foreach ($feeds as $feed) {
|
|
|
|
$md5 = md5($feed->getUrl());
|
|
|
|
$form_definitions[] = [$md5 . '-url', TextType::class, ['data' => $feed->getUrl(), 'label' => ' ']];
|
|
|
|
$form_definitions[] = [$md5 . '-order', IntegerType::class, ['data' => $feed->getOrdering(), 'label' => ' ']];
|
|
|
|
$form_definitions[] = [$md5 . '-title', TextType::class, ['data' => $feed->getTitle(), 'label' => ' ']];
|
|
|
|
$form_definitions[] = [$md5 . '-remove', SubmitType::class, ['label' => _m('Remove')]];
|
|
|
|
}
|
|
|
|
|
|
|
|
$form_definitions[] = ['url', TextType::class, ['label' => _m('New feed'), 'required' => false]];
|
|
|
|
$form_definitions[] = ['order', IntegerType::class, ['label' => _m('Order'), 'data' => (\count($form_definitions) / 4) + 1]];
|
|
|
|
$form_definitions[] = ['title', TextType::class, ['label' => _m('Title'), 'required' => false]];
|
|
|
|
$form_definitions[] = ['add', SubmitType::class, ['label' => _m('Add')]];
|
|
|
|
$form_definitions[] = ['update_exisiting', SubmitType::class, ['label' => _m('Update existing')]];
|
|
|
|
$form_definitions[] = ['reset', SubmitType::class, ['label' => _m('Reset to default values')]];
|
|
|
|
|
|
|
|
$form = Form::create($form_definitions);
|
|
|
|
|
|
|
|
$form->handleRequest($request);
|
|
|
|
if ($form->isSubmitted() && $form->isValid()) {
|
|
|
|
array_pop($form_definitions);
|
|
|
|
array_pop($form_definitions);
|
|
|
|
array_pop($form_definitions);
|
|
|
|
array_pop($form_definitions);
|
|
|
|
array_pop($form_definitions);
|
|
|
|
|
|
|
|
$data = $form->getData();
|
|
|
|
|
|
|
|
if ($form->get('update_exisiting')->isClicked()) {
|
|
|
|
// Each feed has a URL, an order and a title
|
|
|
|
$feeds_data = array_chunk($data, 3, preserve_keys: true);
|
|
|
|
// The last three would be the new one
|
|
|
|
array_pop($feeds_data);
|
|
|
|
// Sort by the order
|
|
|
|
usort($feeds_data, fn ($fd_l, $fd_r) => next($fd_l) <=> next($fd_r));
|
|
|
|
// Make the order sequential
|
|
|
|
$order = 1;
|
|
|
|
foreach ($feeds_data as $i => $fd) {
|
|
|
|
next($fd);
|
|
|
|
$feeds_data[$i][key($fd)] = $order++;
|
|
|
|
}
|
|
|
|
// Update the fields in the corresponding feed
|
|
|
|
foreach ($feeds_data as $fd) {
|
|
|
|
$md5 = str_replace('-url', '', array_key_first($fd));
|
|
|
|
$feed = F\first($feeds, fn ($f) => md5($f->getUrl()) === $md5);
|
|
|
|
$feed->setUrl($fd[$md5 . '-url']);
|
|
|
|
$feed->setOrdering($fd[$md5 . '-order']);
|
|
|
|
$feed->setTitle($fd[$md5 . '-title']);
|
|
|
|
DB::merge($feed);
|
|
|
|
}
|
|
|
|
DB::flush();
|
|
|
|
Cache::delete($key);
|
|
|
|
throw new RedirectException();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Remove feed
|
|
|
|
foreach ($form_definitions as [$field, $type, $opts]) {
|
|
|
|
if (str_ends_with($field, '-url')) {
|
|
|
|
$remove_id = str_replace('-url', '-remove', $field);
|
|
|
|
if ($form->get($remove_id)->isClicked()) {
|
|
|
|
DB::remove(DB::getReference('feed', ['actor_id' => $user->getId(), 'url' => $opts['data']]));
|
|
|
|
DB::flush();
|
|
|
|
Cache::delete($key);
|
|
|
|
throw new RedirectException();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($form->get('reset')->isClicked()) {
|
|
|
|
F\map(DB::findBy('feed', ['actor_id' => $user->getId()]), fn ($f) => DB::remove($f));
|
|
|
|
DB::flush();
|
|
|
|
Cache::delete($key);
|
|
|
|
Feed::createDefaultFeeds($user->getId(), $user);
|
|
|
|
DB::flush();
|
|
|
|
throw new RedirectException();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add feed
|
|
|
|
try {
|
|
|
|
$match = Router::match($data['url']);
|
|
|
|
$route = $match['_route'];
|
|
|
|
DB::persist(Feed::create([
|
|
|
|
'actor_id' => $user->getId(),
|
|
|
|
'url' => $data['url'],
|
|
|
|
'route' => $route,
|
|
|
|
'title' => $data['title'],
|
|
|
|
'ordering' => $data['order'],
|
|
|
|
]));
|
|
|
|
DB::flush();
|
|
|
|
Cache::delete($key);
|
|
|
|
throw new RedirectException();
|
|
|
|
} catch (ResourceNotFoundException) {
|
|
|
|
// throw new ClientException(_m('Invalid route'));
|
|
|
|
// continue bellow
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return [
|
|
|
|
'_template' => 'feeds/edit_feeds.html.twig',
|
|
|
|
'form' => $form->createView(),
|
|
|
|
];
|
|
|
|
}
|
|
|
|
|
2021-04-14 20:54:38 +01:00
|
|
|
public function replies(Request $request)
|
2020-09-05 05:10:33 +01:00
|
|
|
{
|
2021-11-15 17:06:34 +00:00
|
|
|
// TODO replies
|
|
|
|
throw new NotImplementedException;
|
2020-09-06 22:38:37 +01:00
|
|
|
$actor_id = Common::ensureLoggedIn()->getId();
|
2021-10-10 09:26:18 +01:00
|
|
|
$notes = DB::dql('select n from App\Entity\Note n '
|
|
|
|
. 'where n.reply_to is not null and n.actor_id = :id '
|
|
|
|
. 'order by n.created DESC', ['id' => $actor_id], );
|
2020-09-05 05:10:33 +01:00
|
|
|
|
2021-09-06 19:49:03 +01:00
|
|
|
$notes_out = null;
|
2021-08-07 22:52:00 +01:00
|
|
|
Event::handle('FormatNoteList', [$notes, &$notes_out]);
|
2020-12-07 16:23:56 +00:00
|
|
|
|
|
|
|
return [
|
2021-11-26 20:37:17 +00:00
|
|
|
'_template' => 'feeds/feed.html.twig',
|
2021-10-10 09:26:18 +01:00
|
|
|
'notes' => $notes_out,
|
2021-11-26 13:05:23 +00:00
|
|
|
'page_title' => 'Replies feed',
|
2020-12-07 16:23:56 +00:00
|
|
|
];
|
|
|
|
}
|
2020-03-12 17:59:13 +00:00
|
|
|
}
|