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

@ -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 );
}
/**

View File

@ -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';

View File

@ -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;
}
);
}
}

View File

@ -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 );
}
}

View File

@ -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' ) );
}
}

View File

@ -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.

View File

@ -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 );
}
}

View File

@ -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 ) );
}
/**