updated plugin ActivityPub version 8.3.0
This commit is contained in:
@ -30,9 +30,7 @@ class Activity_Object extends Base {
|
||||
return $activity_object;
|
||||
}
|
||||
|
||||
$activity_object = $this->set_audience( $activity_object );
|
||||
|
||||
return $activity_object;
|
||||
return $this->set_audience( $activity_object );
|
||||
}
|
||||
|
||||
/**
|
||||
|
||||
@ -30,8 +30,10 @@ class Attachment extends Post {
|
||||
|
||||
switch ( $mime_type_parts[0] ) {
|
||||
case 'audio':
|
||||
$type = 'Audio';
|
||||
break;
|
||||
case 'video':
|
||||
$type = 'Document';
|
||||
$type = 'Video';
|
||||
break;
|
||||
case 'image':
|
||||
$type = 'Image';
|
||||
|
||||
@ -7,19 +7,25 @@
|
||||
|
||||
namespace Activitypub\Transformer;
|
||||
|
||||
use WP_Comment;
|
||||
use WP_Post;
|
||||
use WP_Term;
|
||||
|
||||
use Activitypub\Activity\Activity;
|
||||
use Activitypub\Collection\Actors;
|
||||
use Activitypub\Activity\Base_Object;
|
||||
use Activitypub\Collection\Actors;
|
||||
use Activitypub\Http;
|
||||
|
||||
use function Activitypub\get_upload_baseurl;
|
||||
use function Activitypub\object_to_uri;
|
||||
|
||||
/**
|
||||
* WordPress Base Transformer.
|
||||
*
|
||||
* Transformers are responsible for transforming a WordPress objects into different ActivityPub
|
||||
* Transformers are responsible for transforming WordPress objects into different ActivityPub
|
||||
* Object-Types or Activities.
|
||||
*
|
||||
* @method string|null get_content() Returns the content for the transformed item.
|
||||
* @method string|array|null get_icon() Returns an icon for the transformed item.
|
||||
* @method string|null get_id() Returns the ID for the transformed item.
|
||||
* @method string|null get_name() Returns the name for the transformed item.
|
||||
* @method string|null get_summary() Returns the summary for the transformed item.
|
||||
*/
|
||||
abstract class Base {
|
||||
/**
|
||||
@ -27,7 +33,7 @@ abstract class Base {
|
||||
*
|
||||
* This is the source object of the transformer.
|
||||
*
|
||||
* @var WP_Post|WP_Comment|Base_Object|string|array|WP_Term
|
||||
* @var \WP_Post|\WP_Comment|Base_Object|string|array|\WP_Term
|
||||
*/
|
||||
protected $item;
|
||||
|
||||
@ -36,7 +42,7 @@ abstract class Base {
|
||||
*
|
||||
* @deprecated version 5.0.0
|
||||
*
|
||||
* @var WP_Post|WP_Comment
|
||||
* @var \WP_Post|\WP_Comment
|
||||
*/
|
||||
protected $wp_object;
|
||||
|
||||
@ -52,7 +58,7 @@ abstract class Base {
|
||||
*
|
||||
* This helps to chain the output of the Transformer.
|
||||
*
|
||||
* @param WP_Post|WP_Comment|Base_Object|string|array|WP_term $item The item that should be transformed.
|
||||
* @param \WP_Post|\WP_Comment|Base_Object|string|array|\WP_term $item The item that should be transformed.
|
||||
*
|
||||
* @return Base
|
||||
*/
|
||||
@ -63,7 +69,7 @@ abstract class Base {
|
||||
/**
|
||||
* Base constructor.
|
||||
*
|
||||
* @param WP_Post|WP_Comment|Base_Object|string|array|WP_Term $item The item that should be transformed.
|
||||
* @param \WP_Post|\WP_Comment|Base_Object|string|array|\WP_Term $item The item that should be transformed.
|
||||
*/
|
||||
public function __construct( $item ) {
|
||||
$this->item = $item;
|
||||
@ -82,6 +88,9 @@ abstract class Base {
|
||||
return $activity_object;
|
||||
}
|
||||
|
||||
// Save activity in the context of an activitypub request.
|
||||
\add_filter( 'activitypub_is_activitypub_request', '__return_true' );
|
||||
|
||||
$vars = $activity_object->get_object_var_keys();
|
||||
|
||||
foreach ( $vars as $var ) {
|
||||
@ -115,13 +124,16 @@ abstract class Base {
|
||||
}
|
||||
}
|
||||
|
||||
// Remove activity in the context of an activitypub request.
|
||||
\remove_filter( 'activitypub_is_activitypub_request', '__return_true' );
|
||||
|
||||
return $activity_object;
|
||||
}
|
||||
|
||||
/**
|
||||
* Transform the item into an ActivityPub Object.
|
||||
*
|
||||
* @return Base_Object|object The Activity-Object.
|
||||
* @return Base_Object The Activity-Object.
|
||||
*/
|
||||
public function to_object() {
|
||||
$activity_object = new Base_Object();
|
||||
@ -131,9 +143,7 @@ abstract class Base {
|
||||
return $activity_object;
|
||||
}
|
||||
|
||||
$activity_object = $this->set_audience( $activity_object );
|
||||
|
||||
return $activity_object;
|
||||
return $this->set_audience( $activity_object );
|
||||
}
|
||||
|
||||
/**
|
||||
@ -168,28 +178,40 @@ abstract class Base {
|
||||
* @return Base_Object The ActivityPub Object.
|
||||
*/
|
||||
protected function set_audience( $activity_object ) {
|
||||
$public = 'https://www.w3.org/ns/activitystreams#Public';
|
||||
$actor = Actors::get_by_resource( $this->get_attributed_to() );
|
||||
if ( ! $actor || is_wp_error( $actor ) ) {
|
||||
$followers = null;
|
||||
} else {
|
||||
$public = 'https://www.w3.org/ns/activitystreams#Public';
|
||||
$followers = null;
|
||||
$replied_to = null;
|
||||
|
||||
$actor = Actors::get_by_resource( $this->get_attributed_to() );
|
||||
if ( ! \is_wp_error( $actor ) ) {
|
||||
$followers = $actor->get_followers();
|
||||
}
|
||||
|
||||
$mentions = array_values( $this->get_mentions() );
|
||||
|
||||
if ( $this->get_in_reply_to() ) {
|
||||
$object = Http::get_remote_object( $this->get_in_reply_to() );
|
||||
if ( $object && ! \is_wp_error( $object ) && isset( $object['attributedTo'] ) ) {
|
||||
$replied_to = array( object_to_uri( $object['attributedTo'] ) );
|
||||
}
|
||||
}
|
||||
|
||||
switch ( $this->get_content_visibility() ) {
|
||||
case ACTIVITYPUB_CONTENT_VISIBILITY_PUBLIC:
|
||||
$activity_object->add_to( $public );
|
||||
$activity_object->add_cc( $followers );
|
||||
$activity_object->add_cc( $mentions );
|
||||
$activity_object->add_cc( $replied_to );
|
||||
break;
|
||||
case ACTIVITYPUB_CONTENT_VISIBILITY_QUIET_PUBLIC:
|
||||
$activity_object->add_to( $followers );
|
||||
$activity_object->add_to( $mentions );
|
||||
$activity_object->add_to( $replied_to );
|
||||
$activity_object->add_cc( $public );
|
||||
break;
|
||||
case ACTIVITYPUB_CONTENT_VISIBILITY_PRIVATE:
|
||||
$activity_object->add_to( $mentions );
|
||||
$activity_object->add_to( $replied_to );
|
||||
}
|
||||
|
||||
return $activity_object;
|
||||
@ -205,6 +227,19 @@ abstract class Base {
|
||||
return $this->get_id();
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns a Tombstone object for the item.
|
||||
*
|
||||
* @return Base_Object The Tombstone object.
|
||||
*/
|
||||
public function to_tombstone() {
|
||||
$object = new Base_Object();
|
||||
$object->set_type( 'Tombstone' );
|
||||
$object->set_id( $this->to_id() );
|
||||
|
||||
return $object;
|
||||
}
|
||||
|
||||
/**
|
||||
* Transforms the ActivityPub Object to an Activity
|
||||
*
|
||||
@ -218,7 +253,7 @@ abstract class Base {
|
||||
$activity = new Activity();
|
||||
$activity->set_type( $type );
|
||||
|
||||
// Pre-fill the Activity with data (for example cc and to).
|
||||
// Pre-fill the Activity with data (for example, cc and to).
|
||||
$activity->set_object( $object );
|
||||
|
||||
// Use simple Object (only ID-URI) for Like and Announce.
|
||||
@ -237,28 +272,6 @@ abstract class Base {
|
||||
protected function get_locale() {
|
||||
$lang = \strtolower( \strtok( \get_locale(), '_-' ) );
|
||||
|
||||
if ( $this->item instanceof \WP_Post ) {
|
||||
/**
|
||||
* Deprecates the `activitypub_post_locale` filter.
|
||||
*
|
||||
* @param string $lang The locale of the post.
|
||||
* @param mixed $item The post object.
|
||||
*
|
||||
* @return string The filtered locale of the post.
|
||||
*/
|
||||
$lang = apply_filters_deprecated(
|
||||
'activitypub_post_locale',
|
||||
array(
|
||||
$lang,
|
||||
$this->item->ID,
|
||||
$this->item,
|
||||
),
|
||||
'5.4.0',
|
||||
'activitypub_locale',
|
||||
'Use the `activitypub_locale` filter instead.'
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* Filter the locale of the post.
|
||||
*
|
||||
@ -372,9 +385,9 @@ abstract class Base {
|
||||
/**
|
||||
* Filter the mentions in the post content.
|
||||
*
|
||||
* @param array $mentions The mentions.
|
||||
* @param string $content The post content.
|
||||
* @param WP_Post $post The post object.
|
||||
* @param array $mentions The mentions.
|
||||
* @param string $content The post content.
|
||||
* @param \WP_Post $post The post object.
|
||||
*
|
||||
* @return array The filtered mentions.
|
||||
*/
|
||||
@ -385,4 +398,339 @@ abstract class Base {
|
||||
$this->item
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns the in reply to.
|
||||
*
|
||||
* @return string|array|null The in reply to.
|
||||
*/
|
||||
protected function get_in_reply_to() {
|
||||
return null;
|
||||
}
|
||||
|
||||
/**
|
||||
* Parse HTML content for image tags and extract attachment information.
|
||||
*
|
||||
* This method is used by both Post and Comment transformers to find images
|
||||
* embedded in HTML content and extract their attachment IDs and alt text.
|
||||
*
|
||||
* @param array $media The existing media array grouped by type.
|
||||
* @param int $max_images Maximum number of images to extract.
|
||||
* @param string $content The HTML content to parse.
|
||||
*
|
||||
* @return array The updated media array with found images.
|
||||
*/
|
||||
protected function parse_html_images( $media, $max_images, $content ) {
|
||||
// If someone calls that function directly, bail.
|
||||
if ( ! \class_exists( '\WP_HTML_Tag_Processor' ) ) {
|
||||
return $media;
|
||||
}
|
||||
|
||||
// Max images can't be negative or zero.
|
||||
if ( $max_images <= 0 ) {
|
||||
return $media;
|
||||
}
|
||||
|
||||
$images = array();
|
||||
$base = get_upload_baseurl();
|
||||
$tags = new \WP_HTML_Tag_Processor( $content );
|
||||
|
||||
// This linter warning is a false positive - we have to re-count each time here as we modify $images.
|
||||
// phpcs:ignore Squiz.PHP.DisallowSizeFunctionsInLoops.Found
|
||||
while ( $tags->next_tag( 'img' ) && ( \count( $images ) <= $max_images ) ) {
|
||||
/**
|
||||
* Filter the image source URL.
|
||||
*
|
||||
* This can be used to modify the image source URL before it is used to
|
||||
* determine the attachment ID.
|
||||
*
|
||||
* @param string $src The image source URL.
|
||||
*/
|
||||
$src = \apply_filters( 'activitypub_image_src', $tags->get_attribute( 'src' ) );
|
||||
|
||||
/*
|
||||
* If the img source is in our uploads dir, get the
|
||||
* associated ID. Note: if there's a -500x500
|
||||
* type suffix, we remove it, but we try the original
|
||||
* first in case the original image is actually called
|
||||
* that. Likewise, we try adding the -scaled suffix for
|
||||
* the case that this is a small version of an image
|
||||
* that was big enough to get scaled down on upload:
|
||||
* https://make.wordpress.org/core/2019/10/09/introducing-handling-of-big-images-in-wordpress-5-3/
|
||||
*/
|
||||
if ( null !== $src && \str_starts_with( $src, $base ) ) {
|
||||
$img_id = \attachment_url_to_postid( $src );
|
||||
|
||||
if ( 0 === $img_id ) {
|
||||
$count = 0;
|
||||
$src = \strtok( $src, '?' );
|
||||
$img_id = \attachment_url_to_postid( $src );
|
||||
}
|
||||
|
||||
if ( 0 === $img_id ) {
|
||||
$count = 0;
|
||||
$src = \preg_replace( '/-(?:\d+x\d+)(\.[a-zA-Z]+)$/', '$1', $src, 1, $count );
|
||||
if ( $count > 0 ) {
|
||||
$img_id = \attachment_url_to_postid( $src );
|
||||
}
|
||||
}
|
||||
|
||||
if ( 0 === $img_id ) {
|
||||
$src = \preg_replace( '/(\.[a-zA-Z]+)$/', '-scaled$1', $src );
|
||||
$img_id = \attachment_url_to_postid( $src );
|
||||
}
|
||||
|
||||
if ( 0 !== $img_id ) {
|
||||
$images[] = array(
|
||||
'id' => $img_id,
|
||||
'alt' => $tags->get_attribute( 'alt' ),
|
||||
);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if ( \count( $media['image'] ) <= $max_images ) {
|
||||
$media['image'] = \array_merge( $media['image'], $images );
|
||||
}
|
||||
|
||||
return $media;
|
||||
}
|
||||
|
||||
/**
|
||||
* Transforms a WordPress attachment array to ActivityStreams attachment format.
|
||||
*
|
||||
* @param array $media The WordPress attachment array with 'id', optional 'alt', and optional 'icon'.
|
||||
*
|
||||
* @return array The ActivityStreams attachment array.
|
||||
*/
|
||||
protected function transform_attachment( $media ) {
|
||||
if ( ! isset( $media['id'] ) ) {
|
||||
return $media;
|
||||
}
|
||||
|
||||
$id = $media['id'];
|
||||
$attachment = array();
|
||||
$mime_type = \get_post_mime_type( $id );
|
||||
$media_type = \strtok( $mime_type, '/' );
|
||||
|
||||
// Switching on image/audio/video.
|
||||
switch ( $media_type ) {
|
||||
case 'image':
|
||||
$image_size = 'large';
|
||||
|
||||
/**
|
||||
* Filter the image URL returned for each post.
|
||||
*
|
||||
* @param array|false $thumbnail The image URL, or false if no image is available.
|
||||
* @param int $id The attachment ID.
|
||||
* @param string $image_size The image size to retrieve. Set to 'large' by default.
|
||||
*/
|
||||
$thumbnail = \apply_filters( 'activitypub_get_image', $this->get_attachment_image_src( $id, $image_size ), $id, $image_size );
|
||||
|
||||
if ( $thumbnail ) {
|
||||
$image = array(
|
||||
'type' => 'Image',
|
||||
'url' => \esc_url( $thumbnail[0] ),
|
||||
'mediaType' => \esc_attr( $mime_type ),
|
||||
);
|
||||
|
||||
if ( ! empty( $media['alt'] ) ) {
|
||||
$image['name'] = \html_entity_decode( \wp_strip_all_tags( $media['alt'] ), ENT_QUOTES, 'UTF-8' );
|
||||
} else {
|
||||
$alt = \get_post_meta( $id, '_wp_attachment_image_alt', true );
|
||||
if ( $alt ) {
|
||||
$image['name'] = \html_entity_decode( \wp_strip_all_tags( $alt ), ENT_QUOTES, 'UTF-8' );
|
||||
}
|
||||
}
|
||||
|
||||
// Add EXIF metadata using Schema.org exifData property (FEP-ee3a).
|
||||
$exif_data = $this->get_exif_data( $id );
|
||||
if ( $exif_data ) {
|
||||
$image['exifData'] = $exif_data;
|
||||
}
|
||||
|
||||
$attachment = $image;
|
||||
}
|
||||
break;
|
||||
|
||||
case 'audio':
|
||||
case 'video':
|
||||
$meta = \wp_get_attachment_metadata( $id );
|
||||
$attachment = array(
|
||||
'type' => \ucfirst( $media_type ),
|
||||
'mediaType' => \esc_attr( $mime_type ),
|
||||
'url' => \esc_url( \wp_get_attachment_url( $id ) ),
|
||||
'name' => \esc_attr( \get_the_title( $id ) ),
|
||||
);
|
||||
|
||||
// Height and width for videos.
|
||||
if ( isset( $meta['width'], $meta['height'] ) ) {
|
||||
$attachment['width'] = \esc_attr( $meta['width'] );
|
||||
$attachment['height'] = \esc_attr( $meta['height'] );
|
||||
}
|
||||
|
||||
// Use poster image from the block, or fall back to the transformer icon.
|
||||
if ( ! empty( $media['icon'] ) ) {
|
||||
$attachment['icon'] = \esc_url_raw( $media['icon'] );
|
||||
} elseif ( \method_exists( $this, 'get_icon' ) && $this->get_icon() ) {
|
||||
$attachment['icon'] = object_to_uri( $this->get_icon() );
|
||||
}
|
||||
break;
|
||||
}
|
||||
|
||||
/**
|
||||
* Filter the attachment for a post.
|
||||
*
|
||||
* @param array $attachment The attachment.
|
||||
* @param int $id The attachment ID.
|
||||
*
|
||||
* @return array The filtered attachment.
|
||||
*/
|
||||
return \apply_filters( 'activitypub_attachment', $attachment, $id );
|
||||
}
|
||||
|
||||
/**
|
||||
* Return details about an image attachment.
|
||||
*
|
||||
* @param int $id The attachment ID.
|
||||
* @param string $image_size The image size to retrieve. Set to 'large' by default.
|
||||
*
|
||||
* @return array|false Array of image data, or boolean false if no image is available.
|
||||
*/
|
||||
protected function get_attachment_image_src( $id, $image_size = 'large' ) {
|
||||
/**
|
||||
* Hook into the image retrieval process. Before image retrieval.
|
||||
*
|
||||
* @param int $id The attachment ID.
|
||||
* @param string $image_size The image size to retrieve. Set to 'large' by default.
|
||||
*/
|
||||
\do_action( 'activitypub_get_image_pre', $id, $image_size );
|
||||
|
||||
$image = \wp_get_attachment_image_src( $id, $image_size );
|
||||
|
||||
/**
|
||||
* Hook into the image retrieval process. After image retrieval.
|
||||
*
|
||||
* @param int $id The attachment ID.
|
||||
* @param string $image_size The image size to retrieve. Set to 'large' by default.
|
||||
*/
|
||||
\do_action( 'activitypub_get_image_post', $id, $image_size );
|
||||
|
||||
return $image;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get EXIF metadata for an image attachment using Schema.org exifData property.
|
||||
*
|
||||
* Returns an array of PropertyValue objects as defined in FEP-ee3a.
|
||||
*
|
||||
* @link https://codeberg.org/fediverse/fep/src/branch/main/fep/ee3a/fep-ee3a.md
|
||||
*
|
||||
* @param int $attachment_id The attachment ID.
|
||||
*
|
||||
* @return array|null Array of PropertyValue objects or null if no EXIF data available.
|
||||
*/
|
||||
protected function get_exif_data( $attachment_id ) {
|
||||
$metadata = \wp_get_attachment_metadata( $attachment_id );
|
||||
|
||||
if ( empty( $metadata['image_meta'] ) ) {
|
||||
return null;
|
||||
}
|
||||
|
||||
$image_meta = $metadata['image_meta'];
|
||||
$exif_data = array();
|
||||
|
||||
// Map WordPress image_meta to FEP-ee3a EXIF field names.
|
||||
if ( ! empty( $image_meta['created_timestamp'] ) ) {
|
||||
$exif_data[] = array(
|
||||
'@type' => 'PropertyValue',
|
||||
'name' => 'DateTime',
|
||||
'value' => \gmdate( 'Y:m:d H:i:s', (int) $image_meta['created_timestamp'] ),
|
||||
);
|
||||
}
|
||||
|
||||
if ( ! empty( $image_meta['shutter_speed'] ) ) {
|
||||
$shutter_speed = (float) $image_meta['shutter_speed'];
|
||||
// Format shutter speed as a fraction (e.g., "1/100") for speeds faster than 1 second.
|
||||
if ( $shutter_speed > 0 && $shutter_speed < 1 ) {
|
||||
$value = '1/' . \round( 1 / $shutter_speed );
|
||||
} elseif ( $shutter_speed >= 1 ) {
|
||||
$value = (string) $shutter_speed;
|
||||
}
|
||||
if ( isset( $value ) ) {
|
||||
$exif_data[] = array(
|
||||
'@type' => 'PropertyValue',
|
||||
'name' => 'ExposureTime',
|
||||
'value' => $value,
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
if ( ! empty( $image_meta['aperture'] ) ) {
|
||||
$exif_data[] = array(
|
||||
'@type' => 'PropertyValue',
|
||||
'name' => 'FNumber',
|
||||
'value' => 'f/' . (float) $image_meta['aperture'],
|
||||
);
|
||||
}
|
||||
|
||||
if ( ! empty( $image_meta['focal_length'] ) ) {
|
||||
$exif_data[] = array(
|
||||
'@type' => 'PropertyValue',
|
||||
'name' => 'FocalLength',
|
||||
'value' => (string) (float) $image_meta['focal_length'],
|
||||
);
|
||||
}
|
||||
|
||||
if ( ! empty( $image_meta['iso'] ) ) {
|
||||
$exif_data[] = array(
|
||||
'@type' => 'PropertyValue',
|
||||
'name' => 'PhotographicSensitivity',
|
||||
'value' => (string) (int) $image_meta['iso'],
|
||||
);
|
||||
}
|
||||
|
||||
if ( ! empty( $image_meta['camera'] ) ) {
|
||||
$exif_data[] = array(
|
||||
'@type' => 'PropertyValue',
|
||||
'name' => 'Model',
|
||||
'value' => \sanitize_text_field( $image_meta['camera'] ),
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* Filter the EXIF data for an image attachment.
|
||||
*
|
||||
* @param array $exif_data Array of PropertyValue objects for Schema.org exifData.
|
||||
* @param array $image_meta The WordPress image_meta array.
|
||||
* @param int $attachment_id The attachment ID.
|
||||
*
|
||||
* @return array The filtered EXIF data array.
|
||||
*/
|
||||
$exif_data = \apply_filters( 'activitypub_image_exif', $exif_data, $image_meta, $attachment_id );
|
||||
|
||||
return ! empty( $exif_data ) ? $exif_data : null;
|
||||
}
|
||||
|
||||
/**
|
||||
* Filter attachments to ensure uniqueness based on their ID.
|
||||
*
|
||||
* @param array $attachments Array of attachments with 'id' field.
|
||||
*
|
||||
* @return array Array with duplicate attachments removed.
|
||||
*/
|
||||
protected function filter_unique_attachments( $attachments ) {
|
||||
$seen_ids = array();
|
||||
|
||||
return \array_filter(
|
||||
$attachments,
|
||||
static function ( $attachment ) use ( &$seen_ids ) {
|
||||
if ( isset( $attachment['id'] ) && ! in_array( $attachment['id'], $seen_ids, true ) ) {
|
||||
$seen_ids[] = $attachment['id'];
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
@ -7,16 +7,17 @@
|
||||
|
||||
namespace Activitypub\Transformer;
|
||||
|
||||
use Activitypub\Webfinger;
|
||||
use Activitypub\Comment as Comment_Utils;
|
||||
use Activitypub\Model\Blog;
|
||||
use Activitypub\Collection\Actors;
|
||||
use Activitypub\Collection\Replies;
|
||||
use Activitypub\Comment as Comment_Utils;
|
||||
use Activitypub\Model\Blog;
|
||||
use Activitypub\Sanitize;
|
||||
use Activitypub\Webfinger;
|
||||
|
||||
use function Activitypub\is_single_user;
|
||||
use function Activitypub\get_rest_url_by_path;
|
||||
use function Activitypub\was_comment_received;
|
||||
use function Activitypub\get_comment_ancestors;
|
||||
use function Activitypub\get_rest_url_by_path;
|
||||
use function Activitypub\is_single_user;
|
||||
use function Activitypub\was_comment_received;
|
||||
|
||||
/**
|
||||
* WordPress Comment Transformer.
|
||||
@ -141,8 +142,8 @@ class Comment extends Base {
|
||||
* @return string The filtered content of the comment.
|
||||
*/
|
||||
$content = \apply_filters( 'comment_text', $content, $comment, array() );
|
||||
$content = \preg_replace( '/[\n\r\t]/', '', $content );
|
||||
$content = \trim( $content );
|
||||
$content = Sanitize::clean_html( $content );
|
||||
$content = Sanitize::strip_whitespace( $content );
|
||||
|
||||
/**
|
||||
* Filter the content of the comment.
|
||||
@ -253,7 +254,7 @@ class Comment extends Base {
|
||||
// Now that we have the full tree of ancestors, only return the ones received from the fediverse.
|
||||
return array_filter(
|
||||
$ancestors,
|
||||
function ( $comment_id ) {
|
||||
static function ( $comment_id ) {
|
||||
return \get_comment_meta( $comment_id, 'protocol', true ) === 'activitypub';
|
||||
}
|
||||
);
|
||||
@ -358,4 +359,61 @@ class Comment extends Base {
|
||||
public function get_replies() {
|
||||
return Replies::get_collection( $this->item );
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the attachment for the comment.
|
||||
*
|
||||
* Extracts images from comment content and returns them as ActivityPub attachments.
|
||||
*
|
||||
* @return array The attachments array for ActivityPub.
|
||||
*/
|
||||
protected function get_attachment() {
|
||||
$max_media = \get_option( 'activitypub_max_image_attachments', \ACTIVITYPUB_MAX_IMAGE_ATTACHMENTS );
|
||||
|
||||
/**
|
||||
* Filters the maximum number of media attachments allowed in a comment.
|
||||
*
|
||||
* @param int $max_media Maximum number of media attachments.
|
||||
* @param \WP_Comment $item The comment object.
|
||||
*/
|
||||
$max_media = (int) \apply_filters( 'activitypub_max_image_attachments', $max_media, $this->item );
|
||||
|
||||
if ( 0 === $max_media ) {
|
||||
return array();
|
||||
}
|
||||
|
||||
$media = array(
|
||||
'image' => array(),
|
||||
'audio' => array(),
|
||||
'video' => array(),
|
||||
);
|
||||
|
||||
// Get comment content and parse for image embeds.
|
||||
$media = $this->parse_html_images( $media, $max_media, $this->item->comment_content );
|
||||
$media = $this->filter_unique_attachments( $media['image'] );
|
||||
$media = \array_slice( $media, 0, $max_media );
|
||||
|
||||
/**
|
||||
* Filter the attachment IDs for a comment.
|
||||
*
|
||||
* @param array $media The media array.
|
||||
* @param \WP_Comment $item The comment object.
|
||||
*
|
||||
* @return array The filtered attachment IDs.
|
||||
*/
|
||||
$media = \apply_filters( 'activitypub_comment_attachment_ids', $media, $this->item );
|
||||
|
||||
// Transform to ActivityStreams format using Base class method.
|
||||
$attachments = \array_filter( \array_map( array( $this, 'transform_attachment' ), $media ) );
|
||||
|
||||
/**
|
||||
* Filter the attachments for a comment.
|
||||
*
|
||||
* @param array $attachments The attachments.
|
||||
* @param \WP_Comment $item The comment object.
|
||||
*
|
||||
* @return array The filtered attachments.
|
||||
*/
|
||||
return \apply_filters( 'activitypub_comment_attachments', $attachments, $this->item );
|
||||
}
|
||||
}
|
||||
|
||||
@ -7,10 +7,11 @@
|
||||
|
||||
namespace Activitypub\Transformer;
|
||||
|
||||
use WP_Error;
|
||||
use Activitypub\Http;
|
||||
use Activitypub\Activity\Base_Object;
|
||||
use Activitypub\Comment as Comment_Helper;
|
||||
use Activitypub\Http;
|
||||
|
||||
use function Activitypub\get_user_id;
|
||||
use function Activitypub\is_post_disabled;
|
||||
use function Activitypub\user_can_activitypub;
|
||||
|
||||
@ -23,9 +24,14 @@ class Factory {
|
||||
*
|
||||
* @param mixed $data The object to transform.
|
||||
*
|
||||
* @return Base|WP_Error The transformer to use, or an error.
|
||||
* @return Base|\WP_Error The transformer to use, or an error.
|
||||
*/
|
||||
public static function get_transformer( $data ) {
|
||||
// Early return for WP_Error objects.
|
||||
if ( \is_wp_error( $data ) ) {
|
||||
return $data;
|
||||
}
|
||||
|
||||
if ( \is_string( $data ) && \filter_var( $data, FILTER_VALIDATE_URL ) ) {
|
||||
$response = Http::get_remote_object( $data );
|
||||
|
||||
@ -40,7 +46,7 @@ class Factory {
|
||||
} elseif ( \is_object( $data ) ) {
|
||||
$class = \get_class( $data );
|
||||
} else {
|
||||
return new WP_Error( 'invalid_object', __( 'Invalid object', 'activitypub' ) );
|
||||
return new \WP_Error( 'invalid_object', __( 'Invalid object', 'activitypub' ) );
|
||||
}
|
||||
|
||||
/**
|
||||
@ -79,7 +85,7 @@ class Factory {
|
||||
! \is_object( $transformer ) ||
|
||||
! $transformer instanceof Base
|
||||
) {
|
||||
return new WP_Error( 'invalid_transformer', __( 'Invalid transformer', 'activitypub' ) );
|
||||
return new \WP_Error( 'invalid_transformer', __( 'Invalid transformer', 'activitypub' ) );
|
||||
}
|
||||
|
||||
return $transformer;
|
||||
@ -90,7 +96,7 @@ class Factory {
|
||||
case 'WP_Post':
|
||||
if ( 'attachment' === $data->post_type && ! is_post_disabled( $data ) ) {
|
||||
return new Attachment( $data );
|
||||
} elseif ( ! is_post_disabled( $data ) ) {
|
||||
} elseif ( ! is_post_disabled( $data ) && get_user_id( $data->post_author ) ) {
|
||||
return new Post( $data );
|
||||
}
|
||||
break;
|
||||
@ -104,14 +110,16 @@ class Factory {
|
||||
return new User( $data );
|
||||
}
|
||||
break;
|
||||
case 'WP_Term':
|
||||
return new Term( $data );
|
||||
case 'json':
|
||||
return new Json( $data );
|
||||
}
|
||||
|
||||
if ( $data instanceof \Activitypub\Activity\Base_Object ) {
|
||||
if ( $data instanceof Base_Object ) {
|
||||
return new Activity_Object( $data );
|
||||
}
|
||||
|
||||
return new WP_Error( 'invalid_object', __( 'Invalid object', 'activitypub' ) );
|
||||
return new \WP_Error( 'invalid_object', __( 'Invalid object', 'activitypub' ) );
|
||||
}
|
||||
}
|
||||
|
||||
@ -7,8 +7,8 @@
|
||||
|
||||
namespace Activitypub\Transformer;
|
||||
|
||||
use function Activitypub\is_actor;
|
||||
use function Activitypub\is_activity;
|
||||
use function Activitypub\is_actor;
|
||||
|
||||
/**
|
||||
* String Transformer Class file.
|
||||
|
||||
File diff suppressed because it is too large
Load Diff
@ -0,0 +1,66 @@
|
||||
<?php
|
||||
/**
|
||||
* Term Transformer Class file.
|
||||
*
|
||||
* @package Activitypub
|
||||
*/
|
||||
|
||||
namespace Activitypub\Transformer;
|
||||
|
||||
/**
|
||||
* Term Transformer Class.
|
||||
*/
|
||||
class Term extends Base {
|
||||
/**
|
||||
* Transforms the WP_Term object to an OrderedCollection.
|
||||
*
|
||||
* @see \Activitypub\Activity\Base_Object
|
||||
*
|
||||
* @return \Activitypub\Activity\Base_Object|\WP_Error The OrderedCollection or WP_Error on failure.
|
||||
*/
|
||||
public function to_object() {
|
||||
$base_object = new \Activitypub\Activity\Base_Object();
|
||||
$base_object->{'@context'} = 'https://www.w3.org/ns/activitystreams';
|
||||
$base_object->set_type( 'OrderedCollection' );
|
||||
$base_object->set_id( $this->get_id() );
|
||||
$base_object->set_url( $this->get_url() );
|
||||
|
||||
return $base_object;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the OrderedCollection ID.
|
||||
*
|
||||
* @return string The OrderedCollection ID.
|
||||
*/
|
||||
public function to_id() {
|
||||
return $this->get_id();
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns the stable ID of the Term.
|
||||
*
|
||||
* Uses term_id query parameter to ensure the ID remains stable
|
||||
* even if the term slug is changed.
|
||||
*
|
||||
* @return string The Term's stable ID.
|
||||
*/
|
||||
public function get_id() {
|
||||
return \add_query_arg( 'term_id', $this->item->term_id, \home_url( '/' ) );
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns the URL of the Term.
|
||||
*
|
||||
* @return string The Term's URL (term link).
|
||||
*/
|
||||
public function get_url() {
|
||||
$term_link = \get_term_link( $this->item );
|
||||
|
||||
if ( \is_wp_error( $term_link ) ) {
|
||||
return '';
|
||||
}
|
||||
|
||||
return \esc_url( $term_link );
|
||||
}
|
||||
}
|
||||
@ -21,13 +21,7 @@ class User extends Base {
|
||||
* @return \Activitypub\Activity\Base_Object|\WP_Error The Actor or WP_Error on failure.
|
||||
*/
|
||||
public function to_object() {
|
||||
$activity_object = $this->transform_object_properties( Actors::get_by_id( $this->item->ID ) );
|
||||
|
||||
if ( \is_wp_error( $activity_object ) ) {
|
||||
return $activity_object;
|
||||
}
|
||||
|
||||
return $activity_object;
|
||||
return $this->transform_object_properties( Actors::get_by_id( $this->item->ID ) );
|
||||
}
|
||||
|
||||
/**
|
||||
|
||||
Reference in New Issue
Block a user