castopod/app/Libraries/ActivityPub/Filters/ActivityPubFilter.php

95 lines
3.0 KiB
PHP

<?php
namespace ActivityPub\Filters;
use Config\Services;
use CodeIgniter\Exceptions\PageNotFoundException;
use Exception;
use ActivityPub\HttpSignature;
use CodeIgniter\HTTP\RequestInterface;
use CodeIgniter\HTTP\ResponseInterface;
use CodeIgniter\Filters\FilterInterface;
use CodeIgniter\HTTP\URI;
class ActivityPubFilter implements FilterInterface
{
/**
* Do whatever processing this filter needs to do.
* By default it should not return anything during
* normal execution. However, when an abnormal state
* is found, it should return an instance of
* CodeIgniter\HTTP\Response. If it does, script
* execution will end and that Response will be
* sent back to the client, allowing for error pages,
* redirects, etc.
*
* @param string[]|null $params
* @return void|mixed
*/
public function before(RequestInterface $request, $params = null)
{
if ($params === null) {
return;
}
if (in_array('verify-activitystream', $params)) {
$negotiate = Services::negotiator();
$allowedContentTypes = [
'application/ld+json; profile="https://www.w3.org/ns/activitystreams',
'application/activity+json',
];
if ($negotiate->media($allowedContentTypes) === '') {
throw PageNotFoundException::forPageNotFound();
}
}
if (in_array('verify-blocks', $params)) {
$payload = $request->getJSON();
$actorUri = $payload->actor;
$domain = (new URI($actorUri))->getHost();
// check first if domain is blocked
if (model('BlockedDomainModel')->isDomainBlocked($domain)) {
throw PageNotFoundException::forPageNotFound();
}
// check if actor is blocked
if (model('ActorModel')->isActorBlocked($actorUri)) {
throw PageNotFoundException::forPageNotFound();
}
}
if (in_array('verify-signature', $params)) {
try {
// securityCheck: check activity signature before handling it
(new HttpSignature())->verify();
} catch (Exception) {
// Invalid HttpSignature (401 = unauthorized)
// TODO: show error message?
return service('response')->setStatusCode(401);
}
}
}
//--------------------------------------------------------------------
/**
* Allows After filters to inspect and modify the response
* object as needed. This method does not allow any way
* to stop execution of other after filters, short of
* throwing an Exception or Error.
*
* @param string[]|null $arguments
*/
public function after(
RequestInterface $request,
ResponseInterface $response,
$arguments = null
): void {
}
//--------------------------------------------------------------------
}