2023-10-22 22:20:53 +00:00
|
|
|
<?php
|
2024-10-09 12:44:17 +00:00
|
|
|
/**
|
|
|
|
* Server REST-Class file.
|
|
|
|
*
|
|
|
|
* @package Activitypub
|
|
|
|
*/
|
|
|
|
|
2023-10-22 22:20:53 +00:00
|
|
|
namespace Activitypub\Rest;
|
|
|
|
|
2023-12-08 23:23:11 +00:00
|
|
|
use WP_Error;
|
2023-10-22 22:20:53 +00:00
|
|
|
use WP_REST_Response;
|
|
|
|
use Activitypub\Signature;
|
2024-06-27 12:10:38 +00:00
|
|
|
use Activitypub\Model\Application;
|
2023-10-22 22:20:53 +00:00
|
|
|
|
|
|
|
/**
|
2024-10-09 12:44:17 +00:00
|
|
|
* ActivityPub Server REST-Class.
|
2023-10-22 22:20:53 +00:00
|
|
|
*
|
|
|
|
* @author Django Doucet
|
|
|
|
*
|
|
|
|
* @see https://www.w3.org/TR/activitypub/#security-verification
|
|
|
|
*/
|
|
|
|
class Server {
|
|
|
|
/**
|
2024-10-09 12:44:17 +00:00
|
|
|
* Initialize the class, registering WordPress hooks.
|
2023-10-22 22:20:53 +00:00
|
|
|
*/
|
|
|
|
public static function init() {
|
|
|
|
self::register_routes();
|
|
|
|
|
2024-10-09 12:44:17 +00:00
|
|
|
\add_filter( 'rest_request_before_callbacks', array( self::class, 'validate_activitypub_requests' ), 9, 3 );
|
2023-10-22 22:20:53 +00:00
|
|
|
\add_filter( 'rest_request_before_callbacks', array( self::class, 'authorize_activitypub_requests' ), 10, 3 );
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Register routes
|
|
|
|
*/
|
|
|
|
public static function register_routes() {
|
|
|
|
\register_rest_route(
|
|
|
|
ACTIVITYPUB_REST_NAMESPACE,
|
|
|
|
'/application',
|
|
|
|
array(
|
|
|
|
array(
|
|
|
|
'methods' => \WP_REST_Server::READABLE,
|
|
|
|
'callback' => array( self::class, 'application_actor' ),
|
|
|
|
'permission_callback' => '__return_true',
|
|
|
|
),
|
|
|
|
)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Render Application actor profile
|
|
|
|
*
|
|
|
|
* @return WP_REST_Response The JSON profile of the Application Actor.
|
|
|
|
*/
|
|
|
|
public static function application_actor() {
|
2024-06-27 12:10:38 +00:00
|
|
|
$user = new Application();
|
2023-10-22 22:20:53 +00:00
|
|
|
|
|
|
|
$json = $user->to_array();
|
|
|
|
|
2023-12-08 23:23:11 +00:00
|
|
|
$rest_response = new WP_REST_Response( $json, 200 );
|
|
|
|
$rest_response->header( 'Content-Type', 'application/activity+json; charset=' . get_option( 'blog_charset' ) );
|
2023-10-22 22:20:53 +00:00
|
|
|
|
2023-12-08 23:23:11 +00:00
|
|
|
return $rest_response;
|
2023-10-22 22:20:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Callback function to authorize each api requests
|
|
|
|
*
|
|
|
|
* @see WP_REST_Request
|
|
|
|
*
|
2024-06-27 12:10:38 +00:00
|
|
|
* @see https://www.w3.org/wiki/SocialCG/ActivityPub/Primer/Authentication_Authorization#Authorized_fetch
|
|
|
|
* @see https://swicg.github.io/activitypub-http-signature/#authorized-fetch
|
|
|
|
*
|
2024-10-09 12:44:17 +00:00
|
|
|
* @param WP_REST_Response|\WP_HTTP_Response|WP_Error|mixed $response Result to send to the client.
|
|
|
|
* Usually a WP_REST_Response or WP_Error.
|
|
|
|
* @param array $handler Route handler used for the request.
|
|
|
|
* @param \WP_REST_Request $request Request used to generate the response.
|
2023-10-22 22:20:53 +00:00
|
|
|
*
|
|
|
|
* @return mixed|WP_Error The response, error, or modified response.
|
|
|
|
*/
|
|
|
|
public static function authorize_activitypub_requests( $response, $handler, $request ) {
|
2023-12-08 23:23:11 +00:00
|
|
|
if ( 'HEAD' === $request->get_method() ) {
|
|
|
|
return $response;
|
|
|
|
}
|
|
|
|
|
2024-10-09 12:44:17 +00:00
|
|
|
if ( \is_wp_error( $response ) ) {
|
|
|
|
return $response;
|
|
|
|
}
|
|
|
|
|
2023-10-22 22:20:53 +00:00
|
|
|
$route = $request->get_route();
|
|
|
|
|
2024-10-09 12:44:17 +00:00
|
|
|
// Check if it is an activitypub request and exclude webfinger and nodeinfo endpoints.
|
2023-10-22 22:20:53 +00:00
|
|
|
if (
|
|
|
|
! \str_starts_with( $route, '/' . ACTIVITYPUB_REST_NAMESPACE ) ||
|
|
|
|
\str_starts_with( $route, '/' . \trailingslashit( ACTIVITYPUB_REST_NAMESPACE ) . 'webfinger' ) ||
|
2024-06-27 12:10:38 +00:00
|
|
|
\str_starts_with( $route, '/' . \trailingslashit( ACTIVITYPUB_REST_NAMESPACE ) . 'nodeinfo' ) ||
|
|
|
|
\str_starts_with( $route, '/' . \trailingslashit( ACTIVITYPUB_REST_NAMESPACE ) . 'application' )
|
2023-10-22 22:20:53 +00:00
|
|
|
) {
|
|
|
|
return $response;
|
|
|
|
}
|
|
|
|
|
2023-12-08 23:23:11 +00:00
|
|
|
/**
|
2024-10-09 12:44:17 +00:00
|
|
|
* Filter to defer signature verification.
|
2023-12-08 23:23:11 +00:00
|
|
|
*
|
|
|
|
* Skip signature verification for debugging purposes or to reduce load for
|
|
|
|
* certain Activity-Types, like "Delete".
|
|
|
|
*
|
2024-10-09 12:44:17 +00:00
|
|
|
* @param bool $defer Whether to defer signature verification.
|
|
|
|
* @param \WP_REST_Request $request The request used to generate the response.
|
2023-12-08 23:23:11 +00:00
|
|
|
*
|
|
|
|
* @return bool Whether to defer signature verification.
|
|
|
|
*/
|
|
|
|
$defer = \apply_filters( 'activitypub_defer_signature_verification', false, $request );
|
|
|
|
|
|
|
|
if ( $defer ) {
|
|
|
|
return $response;
|
|
|
|
}
|
|
|
|
|
2024-06-27 12:10:38 +00:00
|
|
|
if (
|
2024-10-09 12:44:17 +00:00
|
|
|
// POST-Requests are always signed.
|
2024-06-27 12:10:38 +00:00
|
|
|
'GET' !== $request->get_method() ||
|
2024-10-09 12:44:17 +00:00
|
|
|
// GET-Requests only require a signature in secure mode.
|
2024-06-27 12:10:38 +00:00
|
|
|
( 'GET' === $request->get_method() && ACTIVITYPUB_AUTHORIZED_FETCH )
|
|
|
|
) {
|
2023-12-08 23:23:11 +00:00
|
|
|
$verified_request = Signature::verify_http_signature( $request );
|
|
|
|
if ( \is_wp_error( $verified_request ) ) {
|
|
|
|
return new WP_Error(
|
|
|
|
'activitypub_signature_verification',
|
|
|
|
$verified_request->get_error_message(),
|
|
|
|
array( 'status' => 401 )
|
|
|
|
);
|
2023-10-22 22:20:53 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return $response;
|
|
|
|
}
|
2024-10-09 12:44:17 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Callback function to validate incoming ActivityPub requests
|
|
|
|
*
|
|
|
|
* @param WP_REST_Response|\WP_HTTP_Response|WP_Error|mixed $response Result to send to the client.
|
|
|
|
* Usually a WP_REST_Response or WP_Error.
|
|
|
|
* @param array $handler Route handler used for the request.
|
|
|
|
* @param \WP_REST_Request $request Request used to generate the response.
|
|
|
|
*
|
|
|
|
* @return mixed|WP_Error The response, error, or modified response.
|
|
|
|
*/
|
|
|
|
public static function validate_activitypub_requests( $response, $handler, $request ) {
|
|
|
|
if ( 'HEAD' === $request->get_method() ) {
|
|
|
|
return $response;
|
|
|
|
}
|
|
|
|
|
|
|
|
$route = $request->get_route();
|
|
|
|
|
|
|
|
if (
|
|
|
|
\is_wp_error( $response ) ||
|
|
|
|
! \str_starts_with( $route, '/' . ACTIVITYPUB_REST_NAMESPACE )
|
|
|
|
) {
|
|
|
|
return $response;
|
|
|
|
}
|
|
|
|
|
|
|
|
$params = $request->get_json_params();
|
|
|
|
|
|
|
|
// Type is required for ActivityPub requests, so it fail later in the process.
|
|
|
|
if ( ! isset( $params['type'] ) ) {
|
|
|
|
return $response;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (
|
|
|
|
ACTIVITYPUB_DISABLE_INCOMING_INTERACTIONS &&
|
|
|
|
in_array( $params['type'], array( 'Create', 'Like', 'Announce' ), true )
|
|
|
|
) {
|
|
|
|
return new WP_Error(
|
|
|
|
'activitypub_server_does_not_accept_incoming_interactions',
|
|
|
|
\__( 'This server does not accept incoming interactions.', 'activitypub' ),
|
|
|
|
// We have to use a 2XX status code here, because otherwise the response will be
|
|
|
|
// treated as an error and Mastodon might block this WordPress instance.
|
|
|
|
array( 'status' => 202 )
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
return $response;
|
|
|
|
}
|
2023-10-22 22:20:53 +00:00
|
|
|
}
|