. /** * ActivityPub implementation for GNU social * * @package GNUsocial * @author Diogo Cordeiro * @copyright 2018-2019 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 notice representation * * @category Plugin * @package GNUsocial * @author Diogo Cordeiro * @license https://www.gnu.org/licenses/agpl.html GNU AGPL v3 or later */ class Activitypub_notice { /** * Generates a pretty notice from a Notice object * * @param Notice $notice * @return array array to be used in a response * @throws EmptyPkeyValueException * @throws InvalidUrlException * @throws ServerException * @throws Exception * @author Diogo Cordeiro */ public static function notice_to_array($notice) { $profile = $notice->getProfile(); $attachments = []; foreach ($notice->attachments() as $attachment) { $attachments[] = Activitypub_attachment::attachment_to_array($attachment); } $tags = []; foreach ($notice->getTags() as $tag) { if ($tag != "") { // Hacky workaround to avoid stupid outputs $tags[] = Activitypub_tag::tag_to_array($tag); } } if ($notice->isPublic()) { $to = ['https://www.w3.org/ns/activitystreams#Public']; $cc = [common_local_url('apActorFollowers', ['id' => $profile->getID()])]; } else { // Since we currently don't support sending unlisted/followers-only // notices, arriving here means we're instead answering to that type // of posts. Not having subscription policy working, its safer to // always send answers of type unlisted. $to = []; $cc = ['https://www.w3.org/ns/activitystreams#Public']; } foreach ($notice->getAttentionProfiles() as $to_profile) { $to[] = $href = $to_profile->getUri(); $tags[] = Activitypub_mention_tag::mention_tag_to_array_from_values($href, $to_profile->getNickname() . '@' . parse_url($href, PHP_URL_HOST)); } $item = [ '@context' => 'https://www.w3.org/ns/activitystreams', 'id' => self::getUrl($notice), 'type' => 'Note', 'published' => str_replace(' ', 'T', $notice->getCreated()) . 'Z', 'url' => self::getUrl($notice), 'attributedTo' => $profile->getUri(), 'to' => $to, 'cc' => $cc, 'conversation' => $notice->getConversationUrl(), 'content' => $notice->getRendered(), 'isLocal' => $notice->isLocal(), 'attachment' => $attachments, 'tag' => $tags ]; // Is this a reply? if (!empty($notice->reply_to)) { $item['inReplyTo'] = self::getUrl(Notice::getById($notice->reply_to)); } // Do we have a location for this notice? try { $location = Notice_location::locFromStored($notice); $item['latitude'] = $location->lat; $item['longitude'] = $location->lon; } catch (Exception $e) { // Apparently no. } return $item; } /** * Create a Notice via ActivityPub Note Object. * Returns created Notice. * * @param array $object * @param Profile $actor_profile * @param bool $directMessage * @return Notice * @throws Exception * @author Diogo Cordeiro */ public static function create_notice(array $object, Profile $actor_profile = null, bool $directMessage = false): Notice { $id = $object['id']; // int $url = isset($object['url']) ? $object['url'] : $id; // string $content = $object['content']; // string // possible keys: ['inReplyTo', 'latitude', 'longitude'] $settings = []; if (isset($object['inReplyTo'])) { $settings['inReplyTo'] = $object['inReplyTo']; } if (isset($object['latitude'])) { $settings['latitude'] = $object['latitude']; } if (isset($object['longitude'])) { $settings['longitude'] = $object['longitude']; } // Ensure Actor Profile if (is_null($actor_profile)) { if (isset($object['attributedTo'])) { $actor_profile = ActivityPub_explorer::get_profile_from_url($object['attributedTo']); } elseif (isset($object['actor'])) { $actor_profile = ActivityPub_explorer::get_profile_from_url($object['actor']); } else { throw new Exception("A notice can't be created without an actor."); } } $act = new Activity(); $act->verb = ActivityVerb::POST; $act->time = time(); $act->actor = $actor_profile->asActivityObject(); $act->context = new ActivityContext(); $options = ['source' => 'ActivityPub', 'uri' => $id, 'url' => $url, 'is_local' => self::getNotePolicyType($object, $actor_profile)]; if ($directMessage) { $options['scope'] = Notice::MESSAGE_SCOPE; } // Is this a reply? if (isset($settings['inReplyTo'])) { try { $inReplyTo = ActivityPubPlugin::grab_notice_from_url($settings['inReplyTo']); $act->context->replyToID = $inReplyTo->getUri(); $act->context->replyToUrl = $inReplyTo->getUrl(); } catch (Exception $e) { // It failed to grab, maybe we got this note from another source // (e.g.: OStatus) that handles this differently or we really // failed to get it... // Welp, nothing that we can do about, let's // just fake we don't have such notice. } } else { $inReplyTo = null; } // Mentions $mentions = []; if (isset($object['tag']) && is_array($object['tag'])) { foreach ($object['tag'] as $tag) { if ($tag['type'] == 'Mention') { $mentions[] = $tag['href']; } } } $mentions_profiles = []; $discovery = new Activitypub_explorer; foreach ($mentions as $mention) { try { $mentions_profiles[] = $discovery->lookup($mention)[0]; } catch (Exception $e) { // Invalid actor found, just let it go. // TODO: Fallback to OStatus } } unset($discovery); foreach ($mentions_profiles as $mp) { if (!$mp->hasBlocked($actor_profile)) { $act->context->attention[$mp->getUri()] = 'http://activitystrea.ms/schema/1.0/person'; } } // Add location if that is set if (isset($settings['latitude'], $settings['longitude'])) { $act->context->location = Location::fromLatLon($settings['latitude'], $settings['longitude']); } /* Reject notice if it is too long (without the HTML) if (Notice::contentTooLong($content)) { throw new Exception('That\'s too long. Maximum notice size is %d character.'); }*/ $actobj = new ActivityObject(); $actobj->type = ActivityObject::NOTE; $actobj->content = strip_tags($content, '