2024-03-28 09:39:50 +00:00
|
|
|
<?php
|
|
|
|
namespace Activitypub\Integration;
|
|
|
|
|
|
|
|
use DateTime;
|
|
|
|
use Activitypub\Webfinger as Webfinger_Util;
|
2024-05-09 15:26:55 +00:00
|
|
|
use Activitypub\Http;
|
2024-03-28 09:39:50 +00:00
|
|
|
use Activitypub\Collection\Users;
|
|
|
|
use Activitypub\Collection\Followers;
|
|
|
|
use Activitypub\Integration\Nodeinfo;
|
2024-05-09 15:26:55 +00:00
|
|
|
use Enable_Mastodon_Apps\Mastodon_API;
|
2024-03-28 09:39:50 +00:00
|
|
|
use Enable_Mastodon_Apps\Entity\Account;
|
2024-05-09 15:26:55 +00:00
|
|
|
use Enable_Mastodon_Apps\Entity\Status;
|
|
|
|
use Enable_Mastodon_Apps\Entity\Media_Attachment;
|
2024-03-28 09:39:50 +00:00
|
|
|
|
|
|
|
use function Activitypub\get_remote_metadata_by_actor;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Class Enable_Mastodon_Apps
|
|
|
|
*
|
|
|
|
* This class is used to enable Mastodon Apps to work with ActivityPub
|
|
|
|
*
|
|
|
|
* @see https://github.com/akirk/enable-mastodon-apps
|
|
|
|
*/
|
|
|
|
class Enable_Mastodon_Apps {
|
|
|
|
/**
|
|
|
|
* Initialize the class, registering WordPress hooks
|
|
|
|
*/
|
|
|
|
public static function init() {
|
|
|
|
\add_filter( 'mastodon_api_account_followers', array( self::class, 'api_account_followers' ), 10, 2 );
|
2024-05-09 15:26:55 +00:00
|
|
|
\add_filter( 'mastodon_api_account', array( self::class, 'api_account_add_followers' ), 20, 2 );
|
2024-06-27 12:10:38 +00:00
|
|
|
\add_filter( 'mastodon_api_account', array( self::class, 'api_account_external' ), 15, 2 );
|
2024-04-19 10:49:31 +00:00
|
|
|
\add_filter( 'mastodon_api_search', array( self::class, 'api_search' ), 40, 2 );
|
2024-06-27 12:10:38 +00:00
|
|
|
\add_filter( 'mastodon_api_search', array( self::class, 'api_search_by_url' ), 40, 2 );
|
2024-05-09 15:26:55 +00:00
|
|
|
\add_filter( 'mastodon_api_get_posts_query_args', array( self::class, 'api_get_posts_query_args' ) );
|
|
|
|
\add_filter( 'mastodon_api_statuses', array( self::class, 'api_statuses_external' ), 10, 2 );
|
2024-06-27 12:10:38 +00:00
|
|
|
\add_filter( 'mastodon_api_status_context', array( self::class, 'api_get_replies' ), 10, 23 );
|
2024-03-28 09:39:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Add followers to Mastodon API
|
|
|
|
*
|
|
|
|
* @param array $followers An array of followers
|
|
|
|
* @param string $user_id The user id
|
|
|
|
* @param WP_REST_Request $request The request object
|
|
|
|
*
|
|
|
|
* @return array The filtered followers
|
|
|
|
*/
|
|
|
|
public static function api_account_followers( $followers, $user_id ) {
|
|
|
|
$activitypub_followers = Followers::get_followers( $user_id, 40 );
|
|
|
|
$mastodon_followers = array_map(
|
|
|
|
function ( $item ) {
|
|
|
|
$acct = Webfinger_Util::uri_to_acct( $item->get_id() );
|
|
|
|
|
|
|
|
if ( $acct && ! is_wp_error( $acct ) ) {
|
|
|
|
$acct = \str_replace( 'acct:', '', $acct );
|
|
|
|
} else {
|
|
|
|
$acct = $item->get_url();
|
|
|
|
}
|
|
|
|
|
2024-04-19 10:49:31 +00:00
|
|
|
$account = new Account();
|
|
|
|
$account->id = \strval( $item->get__id() );
|
|
|
|
$account->username = $item->get_preferred_username();
|
|
|
|
$account->acct = $acct;
|
|
|
|
$account->display_name = $item->get_name();
|
|
|
|
$account->url = $item->get_url();
|
|
|
|
$account->uri = $item->get_id();
|
|
|
|
$account->avatar = $item->get_icon_url();
|
|
|
|
$account->avatar_static = $item->get_icon_url();
|
|
|
|
$account->created_at = new DateTime( $item->get_published() );
|
|
|
|
$account->last_status_at = new DateTime( $item->get_published() );
|
|
|
|
$account->note = $item->get_summary();
|
|
|
|
$account->header = $item->get_image_url();
|
|
|
|
$account->header_static = $item->get_image_url();
|
|
|
|
$account->followers_count = 0;
|
|
|
|
$account->following_count = 0;
|
|
|
|
$account->statuses_count = 0;
|
|
|
|
$account->bot = false;
|
|
|
|
$account->locked = false;
|
|
|
|
$account->group = false;
|
|
|
|
$account->discoversable = false;
|
|
|
|
$account->indexable = false;
|
|
|
|
$account->hide_collections = false;
|
|
|
|
$account->noindex = false;
|
|
|
|
$account->fields = array();
|
|
|
|
$account->emojis = array();
|
|
|
|
$account->roles = array();
|
|
|
|
|
|
|
|
return $account;
|
2024-03-28 09:39:50 +00:00
|
|
|
},
|
|
|
|
$activitypub_followers
|
|
|
|
);
|
|
|
|
|
|
|
|
$followers = array_merge( $mastodon_followers, $followers );
|
|
|
|
|
|
|
|
return $followers;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Add followers count to Mastodon API
|
|
|
|
*
|
|
|
|
* @param Enable_Mastodon_Apps\Entity\Account $account The account
|
|
|
|
* @param int $user_id The user id
|
|
|
|
*
|
|
|
|
* @return Enable_Mastodon_Apps\Entity\Account The filtered Account
|
|
|
|
*/
|
2024-05-09 15:26:55 +00:00
|
|
|
public static function api_account_add_followers( $account, $user_id ) {
|
2024-06-27 12:10:38 +00:00
|
|
|
if ( ! $account instanceof Account ) {
|
2024-03-28 09:39:50 +00:00
|
|
|
return $account;
|
|
|
|
}
|
|
|
|
|
2024-06-27 12:10:38 +00:00
|
|
|
$user = Users::get_by_various( $user_id );
|
2024-03-28 09:39:50 +00:00
|
|
|
|
|
|
|
if ( ! $user || is_wp_error( $user ) ) {
|
|
|
|
return $account;
|
|
|
|
}
|
|
|
|
|
2024-04-19 10:49:31 +00:00
|
|
|
$header = $user->get_image();
|
|
|
|
if ( $header ) {
|
|
|
|
$account->header = $header['url'];
|
|
|
|
$account->header_static = $header['url'];
|
|
|
|
}
|
|
|
|
|
|
|
|
foreach ( $user->get_attachment() as $attachment ) {
|
|
|
|
if ( 'PropertyValue' === $attachment['type'] ) {
|
|
|
|
$account->fields[] = array(
|
|
|
|
'name' => $attachment['name'],
|
|
|
|
'value' => $attachment['value'],
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-05-09 15:26:55 +00:00
|
|
|
$account->acct = $user->get_preferred_username();
|
2024-04-19 10:49:31 +00:00
|
|
|
$account->note = $user->get_summary();
|
2024-05-09 15:26:55 +00:00
|
|
|
|
2024-06-27 12:10:38 +00:00
|
|
|
$account->followers_count = Followers::count_followers( $user->get__id() );
|
2024-03-28 09:39:50 +00:00
|
|
|
return $account;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Resolve external accounts for Mastodon API
|
|
|
|
*
|
|
|
|
* @param Enable_Mastodon_Apps\Entity\Account $user_data The user data
|
|
|
|
* @param string $user_id The user id
|
|
|
|
*
|
|
|
|
* @return Enable_Mastodon_Apps\Entity\Account The filtered Account
|
|
|
|
*/
|
|
|
|
public static function api_account_external( $user_data, $user_id ) {
|
2024-06-27 12:10:38 +00:00
|
|
|
if ( $user_data || ( is_numeric( $user_id ) && $user_id ) ) {
|
2024-05-09 15:26:55 +00:00
|
|
|
// Only augment.
|
2024-03-28 09:39:50 +00:00
|
|
|
return $user_data;
|
|
|
|
}
|
|
|
|
|
|
|
|
$uri = Webfinger_Util::resolve( $user_id );
|
|
|
|
|
2024-05-09 15:26:55 +00:00
|
|
|
if ( ! $uri || is_wp_error( $uri ) ) {
|
2024-03-28 09:39:50 +00:00
|
|
|
return $user_data;
|
|
|
|
}
|
|
|
|
|
2024-05-09 15:26:55 +00:00
|
|
|
$account = self::get_account_for_actor( $uri );
|
|
|
|
if ( $account ) {
|
|
|
|
return $account;
|
|
|
|
}
|
|
|
|
|
|
|
|
return $user_data;
|
|
|
|
}
|
|
|
|
|
|
|
|
private static function get_account_for_actor( $uri ) {
|
|
|
|
if ( ! is_string( $uri ) ) {
|
|
|
|
return null;
|
|
|
|
}
|
2024-03-28 09:39:50 +00:00
|
|
|
$data = get_remote_metadata_by_actor( $uri );
|
|
|
|
|
|
|
|
if ( ! $data || is_wp_error( $data ) ) {
|
2024-05-09 15:26:55 +00:00
|
|
|
return null;
|
2024-03-28 09:39:50 +00:00
|
|
|
}
|
2024-05-09 15:26:55 +00:00
|
|
|
$account = new Account();
|
2024-03-28 09:39:50 +00:00
|
|
|
|
2024-05-09 15:26:55 +00:00
|
|
|
$acct = Webfinger_Util::uri_to_acct( $uri );
|
|
|
|
if ( str_starts_with( $acct, 'acct:' ) ) {
|
|
|
|
$acct = substr( $acct, 5 );
|
2024-03-28 09:39:50 +00:00
|
|
|
}
|
|
|
|
|
2024-05-09 15:26:55 +00:00
|
|
|
$account->id = $acct;
|
2024-03-28 09:39:50 +00:00
|
|
|
$account->username = $acct;
|
|
|
|
$account->acct = $acct;
|
|
|
|
$account->display_name = $data['name'];
|
|
|
|
$account->url = $uri;
|
2024-05-09 15:26:55 +00:00
|
|
|
|
2024-03-28 09:39:50 +00:00
|
|
|
if ( ! empty( $data['summary'] ) ) {
|
|
|
|
$account->note = $data['summary'];
|
|
|
|
}
|
|
|
|
|
|
|
|
if (
|
|
|
|
isset( $data['icon']['type'] ) &&
|
|
|
|
isset( $data['icon']['url'] ) &&
|
|
|
|
'Image' === $data['icon']['type']
|
|
|
|
) {
|
|
|
|
$account->avatar = $data['icon']['url'];
|
|
|
|
$account->avatar_static = $data['icon']['url'];
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( isset( $data['image'] ) ) {
|
2024-05-09 15:26:55 +00:00
|
|
|
$account->header = $data['image']['url'];
|
|
|
|
$account->header_static = $data['image']['url'];
|
2024-03-28 09:39:50 +00:00
|
|
|
}
|
2024-06-27 12:10:38 +00:00
|
|
|
if ( ! isset( $data['published'] ) ) {
|
|
|
|
$data['published'] = 'now';
|
|
|
|
}
|
2024-03-28 09:39:50 +00:00
|
|
|
$account->created_at = new DateTime( $data['published'] );
|
|
|
|
|
|
|
|
return $account;
|
|
|
|
}
|
2024-04-19 10:49:31 +00:00
|
|
|
|
2024-06-27 12:10:38 +00:00
|
|
|
public static function api_search_by_url( $search_data, $request ) {
|
|
|
|
$p = \wp_parse_url( $request->get_param( 'q' ) );
|
|
|
|
if ( ! $p || ! isset( $p['host'] ) ) {
|
|
|
|
return $search_data;
|
|
|
|
}
|
|
|
|
|
|
|
|
$object = Http::get_remote_object( $request->get_param( 'q' ), true );
|
|
|
|
if ( is_wp_error( $object ) || ! isset( $object['attributedTo'] ) ) {
|
|
|
|
return $search_data;
|
|
|
|
}
|
|
|
|
|
|
|
|
$account = self::get_account_for_actor( $object['attributedTo'] );
|
|
|
|
if ( ! $account ) {
|
|
|
|
return $search_data;
|
|
|
|
}
|
|
|
|
|
|
|
|
$status = self::activity_to_status( $object, $account );
|
|
|
|
if ( $status ) {
|
|
|
|
$search_data['statuses'][] = $status;
|
|
|
|
}
|
|
|
|
|
|
|
|
return $search_data;
|
|
|
|
}
|
|
|
|
|
2024-04-19 10:49:31 +00:00
|
|
|
public static function api_search( $search_data, $request ) {
|
|
|
|
$user_id = \get_current_user_id();
|
|
|
|
if ( ! $user_id ) {
|
|
|
|
return $search_data;
|
|
|
|
}
|
|
|
|
|
|
|
|
$q = $request->get_param( 'q' );
|
|
|
|
if ( ! $q ) {
|
|
|
|
return $search_data;
|
|
|
|
}
|
|
|
|
$q = sanitize_text_field( wp_unslash( $q ) );
|
|
|
|
|
|
|
|
$followers = Followers::get_followers( $user_id, 40, null, array( 's' => $q ) );
|
|
|
|
if ( ! $followers ) {
|
|
|
|
return $search_data;
|
|
|
|
}
|
|
|
|
|
|
|
|
foreach ( $followers as $follower ) {
|
|
|
|
$acct = Webfinger_Util::uri_to_acct( $follower->get_id() );
|
|
|
|
|
|
|
|
if ( $acct && ! is_wp_error( $acct ) ) {
|
|
|
|
$acct = \str_replace( 'acct:', '', $acct );
|
|
|
|
} else {
|
|
|
|
$acct = $follower->get_url();
|
|
|
|
}
|
|
|
|
|
|
|
|
$account = new Account();
|
|
|
|
$account->id = \strval( $follower->get__id() );
|
|
|
|
$account->username = $follower->get_preferred_username();
|
|
|
|
$account->acct = $acct;
|
|
|
|
$account->display_name = $follower->get_name();
|
|
|
|
$account->url = $follower->get_url();
|
|
|
|
$account->uri = $follower->get_id();
|
|
|
|
$account->avatar = $follower->get_icon_url();
|
|
|
|
$account->avatar_static = $follower->get_icon_url();
|
|
|
|
$account->created_at = new DateTime( $follower->get_published() );
|
|
|
|
$account->last_status_at = new DateTime( $follower->get_published() );
|
|
|
|
$account->note = $follower->get_summary();
|
|
|
|
$account->header = $follower->get_image_url();
|
|
|
|
$account->header_static = $follower->get_image_url();
|
|
|
|
|
|
|
|
$search_data['accounts'][] = $account;
|
|
|
|
}
|
|
|
|
|
|
|
|
return $search_data;
|
|
|
|
}
|
2024-05-09 15:26:55 +00:00
|
|
|
|
2024-06-27 12:10:38 +00:00
|
|
|
public static function api_get_posts_query_args( $args ) {
|
2024-05-09 15:26:55 +00:00
|
|
|
if ( isset( $args['author'] ) && is_string( $args['author'] ) ) {
|
|
|
|
$uri = Webfinger_Util::resolve( $args['author'] );
|
|
|
|
if ( $uri && ! is_wp_error( $uri ) ) {
|
|
|
|
$args['activitypub'] = $uri;
|
|
|
|
unset( $args['author'] );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return $args;
|
|
|
|
}
|
|
|
|
|
2024-06-27 12:10:38 +00:00
|
|
|
private static function activity_to_status( $item, $account ) {
|
|
|
|
if ( isset( $item['object'] ) ) {
|
|
|
|
$object = $item['object'];
|
|
|
|
} else {
|
|
|
|
$object = $item;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( ! isset( $object['type'] ) || 'Note' !== $object['type'] ) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
$status = new Status();
|
|
|
|
$status->id = $object['id'];
|
|
|
|
$status->created_at = new DateTime( $object['published'] );
|
|
|
|
$status->content = $object['content'];
|
|
|
|
$status->account = $account;
|
|
|
|
|
|
|
|
if ( ! empty( $object['inReplyTo'] ) ) {
|
|
|
|
$status->in_reply_to_id = $object['inReplyTo'];
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( ! empty( $object['visibility'] ) ) {
|
|
|
|
$status->visibility = $object['visibility'];
|
|
|
|
}
|
|
|
|
if ( ! empty( $object['url'] ) ) {
|
|
|
|
$status->url = $object['url'];
|
|
|
|
$status->uri = $object['url'];
|
|
|
|
} else {
|
|
|
|
$status->uri = $object['id'];
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( ! empty( $object['attachment'] ) ) {
|
|
|
|
$status->media_attachments = array_map(
|
|
|
|
function ( $attachment ) {
|
|
|
|
$default_attachment = array(
|
|
|
|
'url' => null,
|
|
|
|
'mediaType' => null,
|
|
|
|
'name' => null,
|
|
|
|
'width' => 0,
|
|
|
|
'height' => 0,
|
|
|
|
'blurhash' => null,
|
|
|
|
);
|
|
|
|
|
|
|
|
$attachment = array_merge( $default_attachment, $attachment );
|
|
|
|
|
|
|
|
$media_attachment = new Media_Attachment();
|
|
|
|
$media_attachment->id = $attachment['url'];
|
|
|
|
$media_attachment->type = strtok( $attachment['mediaType'], '/' );
|
|
|
|
$media_attachment->url = $attachment['url'];
|
|
|
|
$media_attachment->preview_url = $attachment['url'];
|
|
|
|
$media_attachment->description = $attachment['name'];
|
|
|
|
if ( $attachment['blurhash'] ) {
|
|
|
|
$media_attachment->blurhash = $attachment['blurhash'];
|
|
|
|
}
|
|
|
|
if ( $attachment['width'] > 0 && $attachment['height'] > 0 ) {
|
|
|
|
$media_attachment->meta = array(
|
|
|
|
'original' => array(
|
|
|
|
'width' => $attachment['width'],
|
|
|
|
'height' => $attachment['height'],
|
|
|
|
'size' => $attachment['width'] . 'x' . $attachment['height'],
|
|
|
|
'aspect' => $attachment['width'] / $attachment['height'],
|
|
|
|
),
|
|
|
|
);}
|
|
|
|
return $media_attachment;
|
|
|
|
},
|
|
|
|
$object['attachment']
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
return $status;
|
|
|
|
}
|
|
|
|
|
2024-05-09 15:26:55 +00:00
|
|
|
public static function api_statuses_external( $statuses, $args ) {
|
|
|
|
if ( ! isset( $args['activitypub'] ) ) {
|
|
|
|
return $statuses;
|
|
|
|
}
|
|
|
|
|
|
|
|
$data = get_remote_metadata_by_actor( $args['activitypub'] );
|
|
|
|
|
|
|
|
if ( ! $data || is_wp_error( $data ) || ! isset( $data['outbox'] ) ) {
|
|
|
|
return $statuses;
|
|
|
|
}
|
|
|
|
|
2024-06-27 12:10:38 +00:00
|
|
|
$outbox = Http::get_remote_object( $data['outbox'], true );
|
|
|
|
if ( is_wp_error( $outbox ) || ! isset( $outbox['first'] ) ) {
|
2024-05-09 15:26:55 +00:00
|
|
|
return $statuses;
|
|
|
|
}
|
|
|
|
|
|
|
|
$account = self::get_account_for_actor( $args['activitypub'] );
|
|
|
|
if ( ! $account ) {
|
|
|
|
return $statuses;
|
|
|
|
}
|
2024-06-27 12:10:38 +00:00
|
|
|
$limit = 10;
|
|
|
|
if ( isset( $args['posts_per_page'] ) ) {
|
|
|
|
$limit = $args['posts_per_page'];
|
|
|
|
}
|
|
|
|
if ( $limit > 40 ) {
|
|
|
|
$limit = 40;
|
2024-05-09 15:26:55 +00:00
|
|
|
}
|
2024-06-27 12:10:38 +00:00
|
|
|
$activitypub_statuses = array();
|
|
|
|
$url = $outbox['first'];
|
|
|
|
$tries = 0;
|
|
|
|
while ( $url ) {
|
|
|
|
if ( ++$tries > 3 ) {
|
|
|
|
break;
|
|
|
|
}
|
2024-05-09 15:26:55 +00:00
|
|
|
|
2024-06-27 12:10:38 +00:00
|
|
|
$posts = Http::get_remote_object( $url, true );
|
|
|
|
if ( is_wp_error( $posts ) ) {
|
|
|
|
return $statuses;
|
|
|
|
}
|
2024-05-09 15:26:55 +00:00
|
|
|
|
2024-06-27 12:10:38 +00:00
|
|
|
$new_statuses = array_map(
|
|
|
|
function ( $item ) use ( $account, $args ) {
|
|
|
|
if ( $args['exclude_replies'] ) {
|
|
|
|
if ( isset( $item['object']['inReplyTo'] ) && $item['object']['inReplyTo'] ) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return self::activity_to_status( $item, $account );
|
|
|
|
},
|
|
|
|
$posts['orderedItems']
|
|
|
|
);
|
|
|
|
$activitypub_statuses = array_merge( $activitypub_statuses, array_filter( $new_statuses ) );
|
|
|
|
$url = $posts['next'];
|
|
|
|
|
|
|
|
if ( count( $activitypub_statuses ) >= $limit ) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2024-05-09 15:26:55 +00:00
|
|
|
|
2024-06-27 12:10:38 +00:00
|
|
|
return array_slice( $activitypub_statuses, 0, $limit );
|
|
|
|
}
|
2024-05-09 15:26:55 +00:00
|
|
|
|
2024-06-27 12:10:38 +00:00
|
|
|
public static function api_get_replies( $context, $post_id, $url ) {
|
|
|
|
$meta = Http::get_remote_object( $url, true );
|
|
|
|
if ( is_wp_error( $meta ) || ! isset( $meta['replies']['first']['next'] ) ) {
|
|
|
|
return $context;
|
|
|
|
}
|
2024-05-09 15:26:55 +00:00
|
|
|
|
2024-06-27 12:10:38 +00:00
|
|
|
$replies_url = $meta['replies']['first']['next'];
|
|
|
|
$replies = Http::get_remote_object( $replies_url, true );
|
|
|
|
if ( is_wp_error( $replies ) || ! isset( $replies['items'] ) ) {
|
|
|
|
return $context;
|
|
|
|
}
|
2024-05-09 15:26:55 +00:00
|
|
|
|
2024-06-27 12:10:38 +00:00
|
|
|
foreach ( $replies['items'] as $url ) {
|
|
|
|
$response = Http::get( $url, true );
|
|
|
|
if ( is_wp_error( $response ) || wp_remote_retrieve_response_code( $response ) !== 200 ) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
$status = json_decode( wp_remote_retrieve_body( $response ), true );
|
|
|
|
if ( ! $status || is_wp_error( $status ) ) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
$account = self::get_account_for_actor( $status['attributedTo'] );
|
|
|
|
$status = self::activity_to_status( $status, $account );
|
|
|
|
if ( $status ) {
|
|
|
|
$context['descendants'][ $status->id ] = $status;
|
|
|
|
}
|
|
|
|
}
|
2024-05-09 15:26:55 +00:00
|
|
|
|
2024-06-27 12:10:38 +00:00
|
|
|
return $context;
|
2024-05-09 15:26:55 +00:00
|
|
|
}
|
2024-03-28 09:39:50 +00:00
|
|
|
}
|