[ActivityPub] Fix DELETE
This commit is contained in:
parent
af0366ed58
commit
751b23f6fe
@ -231,12 +231,12 @@ class Notice extends Managed_DataObject
|
|||||||
return $notice;
|
return $notice;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/**
|
||||||
* @param $root boolean If true, link to just the conversation root.
|
* @param bool $anchor If false, link to just the conversation root.
|
||||||
*
|
*
|
||||||
* @return URL to conversation
|
* @return string URL to conversation
|
||||||
*/
|
*/
|
||||||
public function getConversationUrl($anchor=true)
|
public function getConversationUrl(bool $anchor = true): string
|
||||||
{
|
{
|
||||||
return Conversation::getUrlFromNotice($this, $anchor);
|
return Conversation::getUrlFromNotice($this, $anchor);
|
||||||
}
|
}
|
||||||
|
@ -132,7 +132,7 @@ class ActivityPubPlugin extends Plugin
|
|||||||
throw new Exception("The acclaimed actor didn't create this note.");
|
throw new Exception("The acclaimed actor didn't create this note.");
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
throw new Exception("Valid ActivityPub Notice object but unsupported by GNU social.");
|
throw new Exception("Invalid Note Object. Maybe it's a Tombstone?");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -39,6 +39,96 @@ class apNoticeAction extends ManagedAction
|
|||||||
protected $needLogin = false;
|
protected $needLogin = false;
|
||||||
protected $canPost = true;
|
protected $canPost = true;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Notice id
|
||||||
|
* @var int
|
||||||
|
*/
|
||||||
|
public $notice_id;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Notice object to show
|
||||||
|
*/
|
||||||
|
public $notice = null;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Profile of the notice object
|
||||||
|
*/
|
||||||
|
public $profile = null;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Avatar of the profile of the notice object
|
||||||
|
*/
|
||||||
|
public $avatar = null;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Load attributes based on database arguments
|
||||||
|
*
|
||||||
|
* Loads all the DB stuff
|
||||||
|
*
|
||||||
|
* @param array $args $_REQUEST array
|
||||||
|
*
|
||||||
|
* @return bool success flag
|
||||||
|
*/
|
||||||
|
protected function prepare(array $args = []): bool
|
||||||
|
{
|
||||||
|
parent::prepare($args);
|
||||||
|
|
||||||
|
$this->notice_id = (int)$this->trimmed('id');
|
||||||
|
|
||||||
|
try {
|
||||||
|
$this->notice = $this->getNotice();
|
||||||
|
} catch (ClientException $e) {
|
||||||
|
//ActivityPubReturn::error('Activity deleted.', 410);
|
||||||
|
ActivityPubReturn::answer(Activitypub_tombstone::tombstone_to_array(common_local_url('apNotice', ['id' => $this->notice_id])), 410);
|
||||||
|
}
|
||||||
|
$this->target = $this->notice;
|
||||||
|
|
||||||
|
if (!$this->notice->inScope($this->scoped)) {
|
||||||
|
// TRANS: Client exception thrown when trying a view a notice the user has no access to.
|
||||||
|
throw new ClientException(_m('Access restricted.'), 403);
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->profile = $this->notice->getProfile();
|
||||||
|
|
||||||
|
if (!$this->profile instanceof Profile) {
|
||||||
|
// TRANS: Server error displayed trying to show a notice without a connected profile.
|
||||||
|
$this->serverError(_m('Notice has no profile.'), 500);
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
$this->avatar = $this->profile->getAvatar(AVATAR_PROFILE_SIZE);
|
||||||
|
} catch (Exception $e) {
|
||||||
|
$this->avatar = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Is this action read-only?
|
||||||
|
*
|
||||||
|
* @return bool true
|
||||||
|
*/
|
||||||
|
public function isReadOnly($args): bool
|
||||||
|
{
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Last-modified date for page
|
||||||
|
*
|
||||||
|
* When was the content of this page last modified? Based on notice,
|
||||||
|
* profile, avatar.
|
||||||
|
*
|
||||||
|
* @return int last-modified date as unix timestamp
|
||||||
|
*/
|
||||||
|
public function lastModified(): int
|
||||||
|
{
|
||||||
|
return max(strtotime($this->notice->modified),
|
||||||
|
strtotime($this->profile->modified),
|
||||||
|
($this->avatar) ? strtotime($this->avatar->modified) : 0);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Handle the Notice request
|
* Handle the Notice request
|
||||||
*
|
*
|
||||||
@ -48,21 +138,45 @@ class apNoticeAction extends ManagedAction
|
|||||||
* @throws ServerException
|
* @throws ServerException
|
||||||
* @author Diogo Cordeiro <diogo@fc.up.pt>
|
* @author Diogo Cordeiro <diogo@fc.up.pt>
|
||||||
*/
|
*/
|
||||||
protected function handle()
|
protected function handle(): void
|
||||||
{
|
{
|
||||||
try {
|
if (is_null($this->notice)) {
|
||||||
$notice = Notice::getByID($this->trimmed('id'));
|
|
||||||
} catch (Exception $e) {
|
|
||||||
ActivityPubReturn::error('Invalid Activity URI.', 404);
|
ActivityPubReturn::error('Invalid Activity URI.', 404);
|
||||||
}
|
}
|
||||||
|
if (!$this->notice->isLocal()) {
|
||||||
if (!$notice->isLocal()) {
|
|
||||||
// We have no authority on the requested activity.
|
// We have no authority on the requested activity.
|
||||||
ActivityPubReturn::error("This is not a local activity.", 403);
|
ActivityPubReturn::error("This is not a local activity.", 403);
|
||||||
}
|
}
|
||||||
|
|
||||||
$res = Activitypub_notice::notice_to_array($notice);
|
$res = Activitypub_notice::notice_to_array($this->notice);
|
||||||
|
|
||||||
ActivityPubReturn::answer($res);
|
ActivityPubReturn::answer($res);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Fetch the notice to show. This may be overridden by child classes to
|
||||||
|
* customize what we fetch without duplicating all of the prepare() method.
|
||||||
|
*
|
||||||
|
* @return null|Notice null if not found
|
||||||
|
* @throws ClientException If GONE
|
||||||
|
*/
|
||||||
|
protected function getNotice(): ?Notice
|
||||||
|
{
|
||||||
|
$notice = null;
|
||||||
|
try {
|
||||||
|
$notice = Notice::getByID($this->notice_id);
|
||||||
|
// Alright, got it!
|
||||||
|
return $notice;
|
||||||
|
} catch (NoResultException $e) {
|
||||||
|
// Hm, not found.
|
||||||
|
$deleted = null;
|
||||||
|
Event::handle('IsNoticeDeleted', [$this->notice_id, &$deleted]);
|
||||||
|
if ($deleted === true) {
|
||||||
|
// TRANS: Client error displayed trying to show a deleted notice.
|
||||||
|
throw new ClientException(_m('Notice deleted.'), 410);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// No such notice.
|
||||||
|
return null;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -204,7 +204,7 @@ class ActivityPubQueueHandler extends QueueHandler
|
|||||||
*
|
*
|
||||||
* @param $user
|
* @param $user
|
||||||
* @param $notice
|
* @param $notice
|
||||||
* @return boolean hook flag
|
* @return bool hook flag
|
||||||
* @throws HTTP_Request2_Exception
|
* @throws HTTP_Request2_Exception
|
||||||
* @throws InvalidUrlException
|
* @throws InvalidUrlException
|
||||||
* @author Diogo Cordeiro <diogo@fc.up.pt>
|
* @author Diogo Cordeiro <diogo@fc.up.pt>
|
||||||
@ -218,10 +218,6 @@ class ActivityPubQueueHandler extends QueueHandler
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
$other = Activitypub_profile::from_profile_collection(
|
|
||||||
$notice->getAttentionProfiles()
|
|
||||||
);
|
|
||||||
|
|
||||||
if ($notice->reply_to) {
|
if ($notice->reply_to) {
|
||||||
try {
|
try {
|
||||||
$parent_notice = $notice->getParent();
|
$parent_notice = $notice->getParent();
|
||||||
|
@ -229,8 +229,6 @@ class Activitypub_inbox_handler
|
|||||||
/**
|
/**
|
||||||
* Handles a Delete Activity received by our inbox.
|
* Handles a Delete Activity received by our inbox.
|
||||||
*
|
*
|
||||||
* @throws NoProfileException
|
|
||||||
* @throws Exception
|
|
||||||
* @author Bruno Casteleiro <brunoccast@fc.up.pt>
|
* @author Bruno Casteleiro <brunoccast@fc.up.pt>
|
||||||
* @author Diogo Cordeiro <diogo@fc.up.pt>
|
* @author Diogo Cordeiro <diogo@fc.up.pt>
|
||||||
*/
|
*/
|
||||||
@ -240,8 +238,8 @@ class Activitypub_inbox_handler
|
|||||||
if (is_string($object)) {
|
if (is_string($object)) {
|
||||||
$client = new HTTPClient();
|
$client = new HTTPClient();
|
||||||
$response = $client->get($object, ACTIVITYPUB_HTTP_CLIENT_HEADERS);
|
$response = $client->get($object, ACTIVITYPUB_HTTP_CLIENT_HEADERS);
|
||||||
$gone = !$response->isOk();
|
$not_gone = $response->isOk();
|
||||||
if (!$gone) { // It's not gone, we're updating it.
|
if ($not_gone) { // It's not gone, we're updating it.
|
||||||
$object = json_decode($response->getBody(), true);
|
$object = json_decode($response->getBody(), true);
|
||||||
switch ($object['type']) {
|
switch ($object['type']) {
|
||||||
case 'Person':
|
case 'Person':
|
||||||
@ -254,31 +252,33 @@ class Activitypub_inbox_handler
|
|||||||
Activitypub_explorer::get_profile_from_url($object['id']);
|
Activitypub_explorer::get_profile_from_url($object['id']);
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case 'Tombstone':
|
case 'Note': // XXX: We do not support updating a note's contents so, we'll delete and re-fetch for now...
|
||||||
try {
|
try {
|
||||||
$notice = ActivityPubPlugin::grab_notice_from_url($object, false);
|
$notice = ActivityPubPlugin::grab_notice_from_url($object['id'], false);
|
||||||
if ($notice instanceof Notice) {
|
if ($notice instanceof Notice) {
|
||||||
$notice->delete();
|
$notice->delete();
|
||||||
}
|
}
|
||||||
|
ActivityPubPlugin::grab_notice_from_url($object['id'], true);
|
||||||
return;
|
return;
|
||||||
} catch (Exception $e) {
|
} catch (Exception $e) {
|
||||||
// either already deleted or not an object at all
|
// either already deleted or not an object at all
|
||||||
// nothing to do..
|
// nothing to do..
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case 'Note':
|
|
||||||
// XXX: We do not support updating a note's contents so, we'll ignore it for now...
|
|
||||||
default:
|
default:
|
||||||
common_log(LOG_INFO, "Ignoring Delete activity, we do not understand for {$object['type']}.");
|
common_log(LOG_INFO, "Ignoring Delete activity, we do not understand for {$object['type']}.");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else {
|
|
||||||
// We don't know the type of the deleted object :(
|
|
||||||
// Nor if it's gone or not.
|
|
||||||
try {
|
|
||||||
if (is_array($object)) {
|
|
||||||
$object = $object['id'];
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// IFF we reached this point, it either is gone or it's an array
|
||||||
|
// If it's gone, we don't know the type of the deleted object, we only have a Tombstone
|
||||||
|
// If we were given an array, we don't know if it's Gone or not via status code...
|
||||||
|
// In both cases, we will want to fetch the ID and act on that as it is easier than updating the fields
|
||||||
|
|
||||||
|
// Was it a profile?
|
||||||
|
try {
|
||||||
|
$object = $object['id'];
|
||||||
$aprofile = Activitypub_profile::fromUri($object, false);
|
$aprofile = Activitypub_profile::fromUri($object, false);
|
||||||
$res = Activitypub_explorer::get_remote_user_activity($object);
|
$res = Activitypub_explorer::get_remote_user_activity($object);
|
||||||
Activitypub_profile::update_profile($aprofile, $res);
|
Activitypub_profile::update_profile($aprofile, $res);
|
||||||
@ -287,24 +287,22 @@ class Activitypub_inbox_handler
|
|||||||
// Means this wasn't a profile
|
// Means this wasn't a profile
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Was it a note?
|
||||||
try {
|
try {
|
||||||
$client = new HTTPClient();
|
$client = new HTTPClient();
|
||||||
$response = $client->get($object, ACTIVITYPUB_HTTP_CLIENT_HEADERS);
|
/*$response =*/ $client->get($object, ACTIVITYPUB_HTTP_CLIENT_HEADERS);
|
||||||
// If it was deleted
|
// If it was deleted
|
||||||
if ($response->getStatus() == 410) {
|
//if (!$response->isOk()) { // 410 or 404
|
||||||
$notice = ActivityPubPlugin::grab_notice_from_url($object, false);
|
$notice = ActivityPubPlugin::grab_notice_from_url($object, false);
|
||||||
if ($notice instanceof Notice) {
|
if ($notice instanceof Notice) {
|
||||||
$notice->delete();
|
$notice->delete();
|
||||||
}
|
}
|
||||||
} else {
|
// } else
|
||||||
// We can't update a note's contents so, we'll ignore it for now...
|
ActivityPubPlugin::grab_notice_from_url($object, true);
|
||||||
}
|
// XXX: We do not support updating a note's contents so, we'll delete and re-fetch for now...
|
||||||
return;
|
|
||||||
} catch (Exception $e) {
|
} catch (Exception $e) {
|
||||||
// Means we didn't have this note already
|
// Means we didn't have this note already
|
||||||
}
|
// Or we had, deleted and it exploded trying to fetch the Tombstone, either way, we're good.
|
||||||
|
|
||||||
return;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -39,22 +39,13 @@ class Activitypub_delete
|
|||||||
/**
|
/**
|
||||||
* Generates an ActivityPub representation of a Delete
|
* Generates an ActivityPub representation of a Delete
|
||||||
*
|
*
|
||||||
* @param string $actor actor URI
|
* @param Notice $notice
|
||||||
* @param string $object object URI
|
|
||||||
* @return array pretty array to be used in a response
|
* @return array pretty array to be used in a response
|
||||||
* @author Diogo Cordeiro <diogo@fc.up.pt>
|
* @author Diogo Cordeiro <diogo@fc.up.pt>
|
||||||
*/
|
*/
|
||||||
public static function delete_to_array(string $actor, string $object): array
|
public static function delete_to_array(Notice $notice): array
|
||||||
{
|
{
|
||||||
$res = [
|
return Activitypub_notice::notice_to_array($notice);
|
||||||
'@context' => 'https://www.w3.org/ns/activitystreams',
|
|
||||||
'id' => $object . '#delete',
|
|
||||||
'type' => 'Delete',
|
|
||||||
'to' => ['https://www.w3.org/ns/activitystreams#Public'],
|
|
||||||
'actor' => $actor,
|
|
||||||
'object' => $object
|
|
||||||
];
|
|
||||||
return $res;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -43,11 +43,10 @@ class Activitypub_error
|
|||||||
* @param string $m
|
* @param string $m
|
||||||
* @return array pretty array to be used in a response
|
* @return array pretty array to be used in a response
|
||||||
*/
|
*/
|
||||||
public static function error_message_to_array($m)
|
public static function error_message_to_array(string $m): array
|
||||||
{
|
{
|
||||||
$res = [
|
return [
|
||||||
'error'=> $m
|
'error'=> $m
|
||||||
];
|
];
|
||||||
return $res;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -27,7 +27,7 @@
|
|||||||
defined('GNUSOCIAL') || die();
|
defined('GNUSOCIAL') || die();
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* ActivityPub error representation
|
* ActivityPub Like representation
|
||||||
*
|
*
|
||||||
* @category Plugin
|
* @category Plugin
|
||||||
* @package GNUsocial
|
* @package GNUsocial
|
||||||
|
@ -79,6 +79,25 @@ class Activitypub_notice
|
|||||||
$tags[] = Activitypub_mention_tag::mention_tag_to_array_from_values($href, $to_profile->getNickname() . '@' . parse_url($href, PHP_URL_HOST));
|
$tags[] = Activitypub_mention_tag::mention_tag_to_array_from_values($href, $to_profile->getNickname() . '@' . parse_url($href, PHP_URL_HOST));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (ActivityUtils::compareVerbs($notice->getVerb(), ActivityVerb::DELETE)) {
|
||||||
|
$item = [
|
||||||
|
'@context' => 'https://www.w3.org/ns/activitystreams',
|
||||||
|
'id' => self::getUri($notice),
|
||||||
|
'type' => 'Delete',
|
||||||
|
// XXX: A bit of ugly code here
|
||||||
|
'object' => array_merge(Activitypub_tombstone::tombstone_to_array(common_local_url('apNotice', ['id' => (int)substr(explode(':', $notice->getUri())[2],9)])), ['deleted' => str_replace(' ', 'T', $notice->getCreated()) . 'Z']),
|
||||||
|
'url' => $notice->getUrl(),
|
||||||
|
'actor' => $profile->getUri(),
|
||||||
|
'to' => $to,
|
||||||
|
'cc' => $cc,
|
||||||
|
'conversationId' => $notice->getConversationUrl(false),
|
||||||
|
'conversationUrl' => $notice->getConversationUrl(),
|
||||||
|
'content' => $notice->getRendered(),
|
||||||
|
'isLocal' => $notice->isLocal(),
|
||||||
|
'attachment' => $attachments,
|
||||||
|
'tag' => $tags
|
||||||
|
];
|
||||||
|
} else { // Note
|
||||||
$item = [
|
$item = [
|
||||||
'@context' => 'https://www.w3.org/ns/activitystreams',
|
'@context' => 'https://www.w3.org/ns/activitystreams',
|
||||||
'id' => self::getUri($notice),
|
'id' => self::getUri($notice),
|
||||||
@ -88,12 +107,14 @@ class Activitypub_notice
|
|||||||
'attributedTo' => $profile->getUri(),
|
'attributedTo' => $profile->getUri(),
|
||||||
'to' => $to,
|
'to' => $to,
|
||||||
'cc' => $cc,
|
'cc' => $cc,
|
||||||
'conversation' => $notice->getConversationUrl(),
|
'conversationId' => $notice->getConversationUrl(false),
|
||||||
|
'conversationUrl' => $notice->getConversationUrl(),
|
||||||
'content' => $notice->getRendered(),
|
'content' => $notice->getRendered(),
|
||||||
'isLocal' => $notice->isLocal(),
|
'isLocal' => $notice->isLocal(),
|
||||||
'attachment' => $attachments,
|
'attachment' => $attachments,
|
||||||
'tag' => $tags
|
'tag' => $tags
|
||||||
];
|
];
|
||||||
|
}
|
||||||
|
|
||||||
// Is this a reply?
|
// Is this a reply?
|
||||||
if (!empty($notice->reply_to)) {
|
if (!empty($notice->reply_to)) {
|
||||||
|
55
plugins/ActivityPub/lib/models/Activitypub_tombstone.php
Normal file
55
plugins/ActivityPub/lib/models/Activitypub_tombstone.php
Normal file
@ -0,0 +1,55 @@
|
|||||||
|
<?php
|
||||||
|
// This file is part of GNU social - https://www.gnu.org/software/social
|
||||||
|
//
|
||||||
|
// GNU social is free software: you can redistribute it and/or modify
|
||||||
|
// it under the terms of the GNU Affero General Public License as published by
|
||||||
|
// the Free Software Foundation, either version 3 of the License, or
|
||||||
|
// (at your option) any later version.
|
||||||
|
//
|
||||||
|
// GNU social is distributed in the hope that it will be useful,
|
||||||
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
// GNU Affero General Public License for more details.
|
||||||
|
//
|
||||||
|
// You should have received a copy of the GNU Affero General Public License
|
||||||
|
// along with GNU social. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
/**
|
||||||
|
* ActivityPub implementation for GNU social
|
||||||
|
*
|
||||||
|
* @package GNUsocial
|
||||||
|
* @author Diogo Cordeiro <diogo@fc.up.pt>
|
||||||
|
* @copyright 2020 Free Software Foundation, Inc http://www.fsf.org
|
||||||
|
* @license https://www.gnu.org/licenses/agpl.html GNU AGPL v3 or later
|
||||||
|
* @link http://www.gnu.org/software/social/
|
||||||
|
*/
|
||||||
|
|
||||||
|
defined('GNUSOCIAL') || die();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* ActivityPub Tombstone representation
|
||||||
|
*
|
||||||
|
* @category Plugin
|
||||||
|
* @package GNUsocial
|
||||||
|
* @author Diogo Cordeiro <diogo@fc.up.pt>
|
||||||
|
* @license https://www.gnu.org/licenses/agpl.html GNU AGPL v3 or later
|
||||||
|
*/
|
||||||
|
class Activitypub_tombstone
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* Generates an ActivityPub representation of a Tombstone
|
||||||
|
*
|
||||||
|
* @param string $id Activity id
|
||||||
|
* @return array pretty array to be used in a response
|
||||||
|
* @author Diogo Cordeiro <diogo@fc.up.pt>
|
||||||
|
*/
|
||||||
|
public static function tombstone_to_array(string $id): array
|
||||||
|
{
|
||||||
|
$res = [
|
||||||
|
'@context' => 'https://www.w3.org/ns/activitystreams',
|
||||||
|
'id' => $id,
|
||||||
|
'type' => 'Tombstone'
|
||||||
|
];
|
||||||
|
return $res;
|
||||||
|
}
|
||||||
|
}
|
@ -365,10 +365,7 @@ class Activitypub_postman
|
|||||||
*/
|
*/
|
||||||
public function delete_note($notice)
|
public function delete_note($notice)
|
||||||
{
|
{
|
||||||
$data = Activitypub_delete::delete_to_array(
|
$data = Activitypub_delete::delete_to_array($notice);
|
||||||
$notice->getProfile()->getUri(),
|
|
||||||
Activitypub_notice::getUri($notice)
|
|
||||||
);
|
|
||||||
$errors = [];
|
$errors = [];
|
||||||
$data = json_encode($data, JSON_UNESCAPED_SLASHES);
|
$data = json_encode($data, JSON_UNESCAPED_SLASHES);
|
||||||
foreach ($this->to_inbox() as $inbox) {
|
foreach ($this->to_inbox() as $inbox) {
|
||||||
|
Loading…
Reference in New Issue
Block a user