forked from GNUsocial/gnu-social
[ActivityStreamsTwo] Initial Actor support
Various bug fixes
This commit is contained in:
parent
1f3a6fe6ac
commit
365edbaff0
@ -7,6 +7,7 @@ use App\Core\Modules\Plugin;
|
|||||||
use App\Core\Router\RouteLoader;
|
use App\Core\Router\RouteLoader;
|
||||||
use Exception;
|
use Exception;
|
||||||
use Plugin\ActivityPub\Controller\Inbox;
|
use Plugin\ActivityPub\Controller\Inbox;
|
||||||
|
use Plugin\ActivityStreamsTwo\ActivityStreamsTwo;
|
||||||
|
|
||||||
class ActivityPub extends Plugin
|
class ActivityPub extends Plugin
|
||||||
{
|
{
|
||||||
@ -15,13 +16,6 @@ class ActivityPub extends Plugin
|
|||||||
return '3.0.0';
|
return '3.0.0';
|
||||||
}
|
}
|
||||||
|
|
||||||
public static array $accept_headers = [
|
|
||||||
'application/ld+json; profile="https://www.w3.org/ns/activitystreams"',
|
|
||||||
'application/activity+json',
|
|
||||||
'application/json',
|
|
||||||
'application/ld+json',
|
|
||||||
];
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This code executes when GNU social creates the page routing, and we hook
|
* This code executes when GNU social creates the page routing, and we hook
|
||||||
* on this event to add our action handler for Embed.
|
* on this event to add our action handler for Embed.
|
||||||
@ -36,7 +30,7 @@ class ActivityPub extends Plugin
|
|||||||
'activitypub_inbox',
|
'activitypub_inbox',
|
||||||
'{gsactor_id<\d+>}/inbox',
|
'{gsactor_id<\d+>}/inbox',
|
||||||
[Inbox::class, 'handle'],
|
[Inbox::class, 'handle'],
|
||||||
options: ['accept' => self::$accept_headers]
|
options: ['accept' => ActivityStreamsTwo::$accept_headers]
|
||||||
);
|
);
|
||||||
return Event::next;
|
return Event::next;
|
||||||
}
|
}
|
||||||
@ -60,7 +54,7 @@ class ActivityPub extends Plugin
|
|||||||
} elseif (is_array($accept)
|
} elseif (is_array($accept)
|
||||||
&& count(
|
&& count(
|
||||||
array_intersect($accept, self::$accept_headers)
|
array_intersect($accept, self::$accept_headers)
|
||||||
)
|
) > 0
|
||||||
) {
|
) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
@ -5,6 +5,8 @@ namespace Plugin\ActivityStreamsTwo;
|
|||||||
use App\Core\Event;
|
use App\Core\Event;
|
||||||
use App\Core\Modules\Plugin;
|
use App\Core\Modules\Plugin;
|
||||||
use App\Core\Router\RouteLoader;
|
use App\Core\Router\RouteLoader;
|
||||||
|
use Exception;
|
||||||
|
use Plugin\ActivityStreamsTwo\Util\Response\ActorResponse;
|
||||||
use Plugin\ActivityStreamsTwo\Util\Response\NoteResponse;
|
use Plugin\ActivityStreamsTwo\Util\Response\NoteResponse;
|
||||||
use Plugin\ActivityStreamsTwo\Util\Response\TypeResponse;
|
use Plugin\ActivityStreamsTwo\Util\Response\TypeResponse;
|
||||||
|
|
||||||
@ -15,7 +17,7 @@ class ActivityStreamsTwo extends Plugin
|
|||||||
return '0.1.0';
|
return '0.1.0';
|
||||||
}
|
}
|
||||||
|
|
||||||
public array $accept = [
|
public static array $accept_headers = [
|
||||||
'application/ld+json; profile="https://www.w3.org/ns/activitystreams"',
|
'application/ld+json; profile="https://www.w3.org/ns/activitystreams"',
|
||||||
'application/activity+json',
|
'application/activity+json',
|
||||||
'application/json',
|
'application/json',
|
||||||
@ -24,23 +26,28 @@ class ActivityStreamsTwo extends Plugin
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* @param string $route
|
* @param string $route
|
||||||
* @param array $accept
|
* @param array $accept_header
|
||||||
* @param array $vars
|
* @param array $vars
|
||||||
* @param null|TypeResponse $response
|
* @param null|TypeResponse $response
|
||||||
*
|
*
|
||||||
* @throws \Exception
|
*@throws Exception
|
||||||
*
|
*
|
||||||
* @return bool
|
* @return bool
|
||||||
|
*
|
||||||
*/
|
*/
|
||||||
public function onRouteInFormat(string $route, array $accept, array $vars, ?TypeResponse &$response = null): bool
|
public function onControllerResponseInFormat(string $route, array $accept_header, array $vars, ?TypeResponse &$response = null): bool
|
||||||
{
|
{
|
||||||
if (empty(array_intersect($this->accept, $accept))) {
|
if (count(array_intersect(self::$accept_headers, $accept_header)) === 0) {
|
||||||
return Event::next;
|
return Event::next;
|
||||||
}
|
}
|
||||||
switch ($route) {
|
switch ($route) {
|
||||||
case 'note_show':
|
case 'note_view':
|
||||||
$response = NoteResponse::handle($vars['note']);
|
$response = NoteResponse::handle($vars['note']);
|
||||||
return Event::stop;
|
return Event::stop;
|
||||||
|
case 'gsactor_view_id':
|
||||||
|
case 'gsactor_view_nickname':
|
||||||
|
$response = ActorResponse::handle($vars['gsactor']);
|
||||||
|
return Event::stop;
|
||||||
default:
|
default:
|
||||||
return Event::next;
|
return Event::next;
|
||||||
}
|
}
|
||||||
@ -56,11 +63,11 @@ class ActivityStreamsTwo extends Plugin
|
|||||||
*/
|
*/
|
||||||
public function onAddRoute(RouteLoader $r): bool
|
public function onAddRoute(RouteLoader $r): bool
|
||||||
{
|
{
|
||||||
$r->connect('note_view_as2',
|
/*$r->connect('note_view_as2',
|
||||||
'/note/{id<\d+>}',
|
'/note/{id<\d+>}',
|
||||||
[NoteResponse::class, 'handle'],
|
[NoteResponse::class, 'handle'],
|
||||||
options: ['accept' => $this->accept]
|
options: ['accept' => self::$accept_headers]
|
||||||
);
|
);*/
|
||||||
return Event::next;
|
return Event::next;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -0,0 +1,49 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace Plugin\ActivityStreamsTwo\Util\Model\EntityToType;
|
||||||
|
|
||||||
|
use App\Core\Router\Router;
|
||||||
|
use App\Entity\GSActor;
|
||||||
|
use DateTimeInterface;
|
||||||
|
use Exception;
|
||||||
|
use Plugin\ActivityStreamsTwo\Util\Type;
|
||||||
|
|
||||||
|
class GSActorToType
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* @param GSActor $gsactor
|
||||||
|
*
|
||||||
|
* @throws Exception
|
||||||
|
*
|
||||||
|
* @return Type
|
||||||
|
*/
|
||||||
|
public static function translate(GSActor $gsactor)
|
||||||
|
{
|
||||||
|
$uri = Router::url('gsactor_view_id', ['id' => $gsactor->getId()], Router::ABSOLUTE_URL);
|
||||||
|
$attr = [
|
||||||
|
'@context' => 'https://www.w3.org/ns/activitystreams',
|
||||||
|
'id' => $uri,
|
||||||
|
//'inbox' =>
|
||||||
|
//'outbox' =>
|
||||||
|
//'following' =>
|
||||||
|
//'followers' =>
|
||||||
|
//'liked' =>
|
||||||
|
//'streams' =>
|
||||||
|
'preferredUsername' => $gsactor->getNickname(),
|
||||||
|
//'publicKey' => [
|
||||||
|
// 'id' => $uri . "#public-key",
|
||||||
|
// 'owner' => $uri,
|
||||||
|
// 'publicKeyPem' => $public_key
|
||||||
|
// ],
|
||||||
|
'name' => $gsactor->getFullname(),
|
||||||
|
//'icon' =>
|
||||||
|
//'location' =>
|
||||||
|
'published' => $gsactor->getCreated()->format(DateTimeInterface::RFC3339),
|
||||||
|
'summary' => $gsactor->getBio(),
|
||||||
|
//'tag' =>
|
||||||
|
'updated' => $gsactor->getModified()->format(DateTimeInterface::RFC3339),
|
||||||
|
'url' => Router::url('gsactor_view_nickname', ['nickname' => $gsactor->getNickname()], Router::ABSOLUTE_URL),
|
||||||
|
];
|
||||||
|
return Type::create(type: 'Person', attributes: $attr);
|
||||||
|
}
|
||||||
|
}
|
24
plugins/ActivityStreamsTwo/Util/Response/ActorResponse.php
Normal file
24
plugins/ActivityStreamsTwo/Util/Response/ActorResponse.php
Normal file
@ -0,0 +1,24 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace Plugin\ActivityStreamsTwo\Util\Response;
|
||||||
|
|
||||||
|
use App\Entity\GSActor;
|
||||||
|
use Exception;
|
||||||
|
use Plugin\ActivityStreamsTwo\Util\Model\EntityToType\GSActorToType;
|
||||||
|
|
||||||
|
abstract class ActorResponse
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* @param GSActor $gsactor
|
||||||
|
* @param int $status The response status code
|
||||||
|
*
|
||||||
|
* @throws Exception
|
||||||
|
*
|
||||||
|
* @return TypeResponse
|
||||||
|
*/
|
||||||
|
public static function handle(GSActor $gsactor, int $status = 200): TypeResponse
|
||||||
|
{
|
||||||
|
$gsactor->getLocalUser(); // This throws exception if not a local user, which is intended
|
||||||
|
return new TypeResponse(data: GSActorToType::translate($gsactor), status: $status);
|
||||||
|
}
|
||||||
|
}
|
69
src/Controller/GSActor.php
Normal file
69
src/Controller/GSActor.php
Normal file
@ -0,0 +1,69 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
// {{{ License
|
||||||
|
|
||||||
|
// 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/>.
|
||||||
|
|
||||||
|
// }}}
|
||||||
|
|
||||||
|
namespace App\Controller;
|
||||||
|
|
||||||
|
use App\Core\Controller;
|
||||||
|
use App\Core\DB\DB;
|
||||||
|
use function App\Core\I18n\_m;
|
||||||
|
use App\Util\Exception\ClientException;
|
||||||
|
use Symfony\Component\HttpFoundation\Request;
|
||||||
|
|
||||||
|
class GSActor extends Controller
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* Generic function that handles getting a representation for an actor from id
|
||||||
|
*/
|
||||||
|
private function GSActorById(int $id, callable $handle)
|
||||||
|
{
|
||||||
|
$gsactor = DB::findOneBy('gsactor', ['id' => $id]);
|
||||||
|
if (empty($gsactor)) {
|
||||||
|
throw new ClientException(_m('No such actor.'), 404);
|
||||||
|
} else {
|
||||||
|
return $handle($gsactor);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/**
|
||||||
|
* Generic function that handles getting a representation for an actor from nickname
|
||||||
|
*/
|
||||||
|
private function GSActorByNickname(string $nickname, callable $handle)
|
||||||
|
{
|
||||||
|
$user = DB::findOneBy('local_user', ['nickname' => $nickname]);
|
||||||
|
$gsactor = DB::findOneBy('gsactor', ['id' => $user->getId()]);
|
||||||
|
if (empty($gsactor)) {
|
||||||
|
throw new ClientException(_m('No such actor.'), 404);
|
||||||
|
} else {
|
||||||
|
return $handle($gsactor);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The page where the note and it's info is shown
|
||||||
|
*/
|
||||||
|
public function GSActorShowId(Request $request, int $id)
|
||||||
|
{
|
||||||
|
return $this->GSActorById($id, fn ($gsactor) => ['_template' => 'actor/view.html.twig', 'gsactor' => $gsactor]);
|
||||||
|
}
|
||||||
|
public function GSActorShowNickname(Request $request, string $nickname)
|
||||||
|
{
|
||||||
|
return $this->GSActorByNickname($nickname, fn ($gsactor) => ['_template' => 'actor/view.html.twig', 'gsactor' => $gsactor]);
|
||||||
|
}
|
||||||
|
}
|
@ -36,7 +36,7 @@ class Note extends Controller
|
|||||||
{
|
{
|
||||||
$note = DB::findOneBy('note', ['id' => $id]);
|
$note = DB::findOneBy('note', ['id' => $id]);
|
||||||
if (empty($note)) {
|
if (empty($note)) {
|
||||||
throw new ClientException(_m('No such note'), 404);
|
throw new ClientException(_m('No such note.'), 404);
|
||||||
} else {
|
} else {
|
||||||
return $handle($note);
|
return $handle($note);
|
||||||
}
|
}
|
||||||
@ -45,7 +45,7 @@ class Note extends Controller
|
|||||||
/**
|
/**
|
||||||
* The page where the note and it's info is shown
|
* The page where the note and it's info is shown
|
||||||
*/
|
*/
|
||||||
public function note_show(Request $request, int $id)
|
public function NoteShow(Request $request, int $id)
|
||||||
{
|
{
|
||||||
return $this->note($id, fn ($note) => ['_template' => 'note/view.html.twig', 'note' => $note]);
|
return $this->note($id, fn ($note) => ['_template' => 'note/view.html.twig', 'note' => $note]);
|
||||||
}
|
}
|
||||||
|
@ -126,7 +126,7 @@ class Controller extends AbstractController implements EventSubscriberInterface
|
|||||||
'accept_header' => $accept,
|
'accept_header' => $accept,
|
||||||
'vars' => $this->vars,
|
'vars' => $this->vars,
|
||||||
'response' => &$potential_response,
|
'response' => &$potential_response,
|
||||||
]) === Event::next) {
|
]) !== Event::stop) {
|
||||||
switch ($format) {
|
switch ($format) {
|
||||||
case 'html':
|
case 'html':
|
||||||
$event->setResponse($this->render($template, $this->vars));
|
$event->setResponse($this->render($template, $this->vars));
|
||||||
|
@ -49,7 +49,6 @@ class GSActor extends Entity
|
|||||||
// @codeCoverageIgnoreStart
|
// @codeCoverageIgnoreStart
|
||||||
private int $id;
|
private int $id;
|
||||||
private string $nickname;
|
private string $nickname;
|
||||||
private string $normalized_nickname;
|
|
||||||
private ?string $fullname;
|
private ?string $fullname;
|
||||||
private int $roles = 4;
|
private int $roles = 4;
|
||||||
private ?string $homepage;
|
private ?string $homepage;
|
||||||
@ -84,17 +83,6 @@ class GSActor extends Entity
|
|||||||
return $this->nickname;
|
return $this->nickname;
|
||||||
}
|
}
|
||||||
|
|
||||||
public function setNormalizedNickname(string $normalized_nickname): self
|
|
||||||
{
|
|
||||||
$this->normalized_nickname = $normalized_nickname;
|
|
||||||
return $this;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function getNormalizedNickname(): string
|
|
||||||
{
|
|
||||||
return $this->normalized_nickname;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function setFullname(?string $fullname): self
|
public function setFullname(?string $fullname): self
|
||||||
{
|
{
|
||||||
$this->fullname = $fullname;
|
$this->fullname = $fullname;
|
||||||
@ -219,6 +207,11 @@ class GSActor extends Entity
|
|||||||
// @codeCoverageIgnoreEnd
|
// @codeCoverageIgnoreEnd
|
||||||
// }}} Autocode
|
// }}} Autocode
|
||||||
|
|
||||||
|
public function getLocalUser()
|
||||||
|
{
|
||||||
|
return DB::findOneBy('local_user', ['id' => $this->getId()]);
|
||||||
|
}
|
||||||
|
|
||||||
public function getAvatarUrl()
|
public function getAvatarUrl()
|
||||||
{
|
{
|
||||||
$url = null;
|
$url = null;
|
||||||
|
47
src/Routes/GSActor.php
Normal file
47
src/Routes/GSActor.php
Normal file
@ -0,0 +1,47 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
// {{{ License
|
||||||
|
|
||||||
|
// 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/>.
|
||||||
|
|
||||||
|
// }}}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Define social's attachment routes
|
||||||
|
*
|
||||||
|
* @package GNUsocial
|
||||||
|
* @category Router
|
||||||
|
*
|
||||||
|
* @author Diogo Cordeiro <mail@diogo.site>
|
||||||
|
* @author Hugo Sales <hugo@hsal.es>
|
||||||
|
* @copyright 2020-2021 Free Software Foundation, Inc http://www.fsf.org
|
||||||
|
* @license https://www.gnu.org/licenses/agpl.html GNU AGPL v3 or later
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace App\Routes;
|
||||||
|
|
||||||
|
use App\Controller as C;
|
||||||
|
use App\Core\Router\RouteLoader;
|
||||||
|
use App\Util\Nickname;
|
||||||
|
|
||||||
|
abstract class GSActor
|
||||||
|
{
|
||||||
|
public static function load(RouteLoader $r): void
|
||||||
|
{
|
||||||
|
$r->connect(id: 'gsactor_view_id', uri_path: '/actor/{id<\d+>}', target: [C\GSActor::class, 'GSActorShowId']);
|
||||||
|
$r->connect(id: 'gsactor_view_nickname', uri_path: '/{nickname<' . Nickname::DISPLAY_FMT . '>}', target: [C\GSActor::class, 'GSActorShowNickname']);
|
||||||
|
}
|
||||||
|
}
|
@ -40,6 +40,6 @@ abstract class Note
|
|||||||
{
|
{
|
||||||
public static function load(RouteLoader $r): void
|
public static function load(RouteLoader $r): void
|
||||||
{
|
{
|
||||||
$r->connect('note_view', '/note/{id<\d+>}', [C\Note::class, 'note_show']);
|
$r->connect('note_view', '/note/{id<\d+>}', [C\Note::class, 'NoteShow']);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user