updated plugin ActivityPub version 8.3.0

This commit is contained in:
2026-06-03 21:28:46 +00:00
committed by Gitium
parent a4b78ec277
commit 6fe182458a
340 changed files with 43232 additions and 7568 deletions

View File

@ -0,0 +1,336 @@
<?php
/**
* Handler for QuoteRequest activities.
*
* @package Activitypub
*/
namespace Activitypub\Handler;
use Activitypub\Activity\Activity;
use Activitypub\Collection\Actors;
use Activitypub\Collection\Followers;
use Activitypub\Collection\Inbox;
use Activitypub\Collection\Remote_Actors;
use function Activitypub\add_to_outbox;
use function Activitypub\object_to_uri;
use function Activitypub\user_can_activitypub;
/**
* Handler for QuoteRequest activities.
*
* @see https://codeberg.org/fediverse/fep/src/branch/main/fep/044f/fep-044f.md
*/
class Quote_Request {
/**
* Initialize the class, registering WordPress hooks.
*/
public static function init() {
\add_action( 'activitypub_inbox_quote_request', array( self::class, 'handle_quote_request' ), 10, 2 );
\add_action( 'activitypub_rest_inbox_disallowed', array( self::class, 'handle_blocked_request' ), 10, 3 );
\add_action( 'delete_comment', array( self::class, 'handle_quote_delete' ), 10, 2 );
\add_filter( 'activitypub_validate_object', array( self::class, 'validate_object' ), 10, 3 );
}
/**
* Handle QuoteRequest activities.
*
* @param array $activity The activity object.
* @param int|int[] $user_ids The user ID(s).
*/
public static function handle_quote_request( $activity, $user_ids ) {
$state = true;
$post_id = \url_to_postid( object_to_uri( $activity['object'] ) );
$post = $post_id ? \get_post( $post_id ) : null;
if ( ! $post ) {
$user_id = \is_array( $user_ids ) ? \reset( $user_ids ) : $user_ids;
self::queue_reject( $activity, $user_id );
return;
}
// Use the post author as the responding actor — they own the quoted content.
$user_id = (int) $post->post_author;
$content_policy = \get_post_meta( $post_id, 'activitypub_interaction_policy_quote', true );
// Fall back to global default if not set.
if ( ! $content_policy ) {
$content_policy = \get_option( 'activitypub_default_quote_policy', ACTIVITYPUB_INTERACTION_POLICY_ANYONE );
}
switch ( $content_policy ) {
case ACTIVITYPUB_INTERACTION_POLICY_ME:
self::queue_reject( $activity, $user_id );
$state = false;
break;
case ACTIVITYPUB_INTERACTION_POLICY_FOLLOWERS:
$follower = Remote_Actors::get_by_uri( object_to_uri( $activity['actor'] ) );
if ( ! \is_wp_error( $follower ) && Followers::follows( $follower->ID, $user_id ) ) {
self::queue_accept( $activity, $user_id, $post_id );
} else {
self::queue_reject( $activity, $user_id );
$state = false;
}
break;
case ACTIVITYPUB_INTERACTION_POLICY_ANYONE:
default:
self::queue_accept( $activity, $user_id, $post_id );
break;
}
/**
* Fires after an ActivityPub QuoteRequest activity has been handled.
*
* @param array $activity The ActivityPub activity data.
* @param int[] $user_ids The local user IDs.
* @param bool $success True on success, false otherwise.
* @param string $content_policy The content policy for the quoted post.
*/
\do_action( 'activitypub_handled_quote_request', $activity, (array) $user_ids, $state, $content_policy );
}
/**
* ActivityPub inbox disallowed activity.
*
* @param array $activity The activity array.
* @param int|int[]|null $user_ids The user ID(s).
* @param string $type The type of the activity.
*/
public static function handle_blocked_request( $activity, $user_ids, $type ) {
if ( ! in_array( strtolower( $type ), array( 'quoterequest', 'quote_request' ), true ) ) {
return;
}
// Extract the user ID (quote requests are always for a single user).
$user_id = \is_array( $user_ids ) ? \reset( $user_ids ) : $user_ids;
self::queue_reject( $activity, $user_id );
}
/**
* Handle deletion of a quote comment.
*
* When a local quote comment is deleted, send a Reject activity to revoke
* the previously accepted QuoteRequest.
*
* @param int $comment_id The comment ID being deleted.
* @param \WP_Comment|null $comment The comment object, or null if not available.
*/
public static function handle_quote_delete( $comment_id, $comment ) {
// Try to get comment if not provided.
if ( ! $comment ) {
$comment = \get_comment( $comment_id );
}
// Only handle quote comments.
if ( ! $comment || 'quote' !== $comment->comment_type ) {
return;
}
// Get the post being quoted.
$post_id = $comment->comment_post_ID;
if ( ! $post_id ) {
return;
}
// Get the instrument URL (the quote post URL) from comment meta.
$instrument_url = \get_comment_meta( $comment_id, 'source_url', true );
if ( ! $instrument_url ) {
$instrument_url = \get_comment_meta( $comment_id, 'source_id', true );
}
if ( ! $instrument_url ) {
return;
}
// Get the post author (who accepted the quote).
$post = \get_post( $post_id );
if ( ! $post || ! $post->post_author ) {
return;
}
/*
* Try to retrieve the original QuoteRequest from the inbox.
* For QuoteRequest activities, the inbox stores the instrument URL
* in _activitypub_object_id, so we can query by that.
*/
$activity_object = null;
$inbox_item = Inbox::get_by_type_and_object( 'QuoteRequest', $instrument_url );
if ( $inbox_item instanceof \WP_Post ) {
$activity_object = \json_decode( $inbox_item->post_content, true );
if ( JSON_ERROR_NONE !== \json_last_error() ) {
$activity_object = null;
}
}
// Fallback: If inbox item not found, reconstruct from available data.
if ( ! $activity_object ) {
$activity_object = array(
'type' => 'QuoteRequest',
'actor' => $comment->comment_author_url,
'object' => \get_permalink( $post_id ),
'instrument' => $instrument_url,
'published' => \gmdate( 'c' ),
);
}
// Remove from _activitypub_quoted_by meta.
\delete_post_meta( $post_id, '_activitypub_quoted_by', $instrument_url );
// Send Reject activity to revoke the quote permission.
self::queue_reject( $activity_object, $post->post_author );
/**
* Fires after a quote comment has been deleted and Reject activity sent.
*
* @param int $comment_id The deleted comment ID.
* @param int $post_id The post ID that was quoted.
* @param string $instrument_url The instrument URL (quote post).
* @param array $activity_object The QuoteRequest activity that was rejected.
*/
\do_action( 'activitypub_quote_comment_deleted', $comment_id, $post_id, $instrument_url, $activity_object );
}
/**
* Send an Accept activity in response to the QuoteRequest.
*
* @see https://codeberg.org/fediverse/fep/src/branch/main/fep/044f/fep-044f.md#example-accept
*
* @param array $activity_object The activity object.
* @param int $user_id The user ID.
* @param int $post_id The post ID.
*/
public static function queue_accept( $activity_object, $user_id, $post_id ) {
// Fall back to the blog actor if the user has ActivityPub disabled.
if ( ! user_can_activitypub( $user_id ) ) {
$user_id = Actors::BLOG_USER_ID;
}
$actor = Actors::get_by_id( $user_id );
if ( \is_wp_error( $actor ) ) {
return;
}
$activity_object['instrument'] = object_to_uri( $activity_object['instrument'] );
$post_meta = \get_post_meta( $post_id, '_activitypub_quoted_by', false );
if ( in_array( $activity_object['instrument'], $post_meta, true ) ) {
global $wpdb;
// phpcs:ignore WordPress.DB.DirectDatabaseQuery.DirectQuery, WordPress.DB.DirectDatabaseQuery.NoCaching
$meta_id = $wpdb->get_var(
$wpdb->prepare(
"SELECT meta_id FROM {$wpdb->postmeta} WHERE post_id = %d AND meta_key = %s AND meta_value = %s LIMIT 1",
$post_id,
'_activitypub_quoted_by',
$activity_object['instrument']
)
);
} else {
$meta_id = \add_post_meta( $post_id, '_activitypub_quoted_by', $activity_object['instrument'] );
}
// Only send minimal data.
$activity_object = array_intersect_key(
$activity_object,
array(
'id' => 1,
'type' => 1,
'actor' => 1,
'object' => 1,
'instrument' => 1,
)
);
$url = \add_query_arg(
array(
'p' => $post_id,
'stamp' => $meta_id,
),
\home_url( '/' )
);
$activity = new Activity();
$activity->set_type( 'Accept' );
$activity->set_actor( $actor->get_id() );
$activity->set_object( $activity_object );
$activity->set_result( $url );
$activity->add_to( object_to_uri( $activity_object['actor'] ) );
add_to_outbox( $activity, null, $user_id, ACTIVITYPUB_CONTENT_VISIBILITY_PRIVATE );
}
/**
* Send a Reject activity in response to the QuoteRequest.
*
* @see https://codeberg.org/fediverse/fep/src/branch/main/fep/044f/fep-044f.md#example-reject
*
* @param array $activity_object The activity object.
* @param int $user_id The user ID.
*/
public static function queue_reject( $activity_object, $user_id ) {
// Fall back to the blog actor if the user has ActivityPub disabled.
if ( ! user_can_activitypub( $user_id ) ) {
$user_id = Actors::BLOG_USER_ID;
}
$actor = Actors::get_by_id( $user_id );
if ( \is_wp_error( $actor ) ) {
return;
}
$activity_object['instrument'] = object_to_uri( $activity_object['instrument'] );
// Only send minimal data.
$activity_object = array_intersect_key(
$activity_object,
array(
'id' => 1,
'type' => 1,
'actor' => 1,
'object' => 1,
'instrument' => 1,
)
);
$activity = new Activity();
$activity->set_type( 'Reject' );
$activity->set_actor( $actor->get_id() );
$activity->set_object( $activity_object );
$activity->add_to( object_to_uri( $activity_object['actor'] ) );
add_to_outbox( $activity, null, $user_id, ACTIVITYPUB_CONTENT_VISIBILITY_PRIVATE );
}
/**
* Validate the object.
*
* @param bool $valid The validation state.
* @param string $param The object parameter.
* @param \WP_REST_Request $request The request object.
*
* @return bool The validation state: true if valid, false if not.
*/
public static function validate_object( $valid, $param, $request ) {
$activity = $request->get_json_params();
if ( empty( $activity['type'] ) ) {
return false;
}
if ( 'QuoteRequest' !== $activity['type'] ) {
return $valid;
}
if ( ! isset( $activity['actor'], $activity['object'], $activity['instrument'] ) ) {
return false;
}
return $valid;
}
}