2018-02-07 08:31:17 -05:00
|
|
|
<?php
|
2020-02-09 09:45:36 -05:00
|
|
|
/**
|
2022-01-02 02:27:47 -05:00
|
|
|
* @copyright Copyright (C) 2010-2022, the Friendica project
|
2020-02-09 09:45:36 -05:00
|
|
|
*
|
|
|
|
* @license GNU AGPL version 3 or any later version
|
|
|
|
*
|
|
|
|
* This program 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.
|
|
|
|
*
|
|
|
|
* This program 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 this program. If not, see <https://www.gnu.org/licenses/>.
|
|
|
|
*
|
2018-02-07 08:31:17 -05:00
|
|
|
*/
|
2020-02-09 09:45:36 -05:00
|
|
|
|
2018-02-07 08:31:17 -05:00
|
|
|
namespace Friendica\Core;
|
|
|
|
|
2021-11-07 01:57:18 -04:00
|
|
|
use Friendica\Database\DBA;
|
2022-09-12 17:12:11 -04:00
|
|
|
use Friendica\Model\Item;
|
2021-11-07 01:57:18 -04:00
|
|
|
use Friendica\Model\User;
|
2021-09-26 10:30:44 -04:00
|
|
|
use Friendica\Network\HTTPException;
|
|
|
|
use Friendica\Protocol\Activity;
|
|
|
|
use Friendica\Protocol\ActivityPub;
|
|
|
|
use Friendica\Protocol\Diaspora;
|
|
|
|
use Friendica\Protocol\OStatus;
|
|
|
|
use Friendica\Protocol\Salmon;
|
2020-03-04 17:28:41 -05:00
|
|
|
|
2018-02-07 08:31:17 -05:00
|
|
|
/**
|
|
|
|
* Manage compatibility with federated networks
|
|
|
|
*/
|
|
|
|
class Protocol
|
|
|
|
{
|
2018-08-30 17:47:48 -04:00
|
|
|
// Native support
|
2019-01-16 16:39:56 -05:00
|
|
|
const ACTIVITYPUB = 'apub'; // ActivityPub (Pleroma, Mastodon, Osada, ...)
|
2018-08-30 17:47:48 -04:00
|
|
|
const DFRN = 'dfrn'; // Friendica, Mistpark, other DFRN implementations
|
2019-01-16 16:39:56 -05:00
|
|
|
const DIASPORA = 'dspr'; // Diaspora, Hubzilla, Socialhome, Ganggo
|
2018-08-30 17:47:48 -04:00
|
|
|
const FEED = 'feed'; // RSS/Atom feeds with no known "post/notify" protocol
|
|
|
|
const MAIL = 'mail'; // IMAP/POP
|
2019-01-16 16:39:56 -05:00
|
|
|
const OSTATUS = 'stat'; // GNU Social and other OStatus implementations
|
2018-08-30 17:47:48 -04:00
|
|
|
|
|
|
|
const NATIVE_SUPPORT = [self::DFRN, self::DIASPORA, self::OSTATUS, self::FEED, self::MAIL, self::ACTIVITYPUB];
|
2018-02-07 08:31:17 -05:00
|
|
|
|
2019-07-01 14:00:55 -04:00
|
|
|
const FEDERATED = [self::DFRN, self::DIASPORA, self::OSTATUS, self::ACTIVITYPUB];
|
|
|
|
|
2019-07-19 22:19:29 -04:00
|
|
|
const SUPPORT_PRIVATE = [self::DFRN, self::DIASPORA, self::MAIL, self::ACTIVITYPUB, self::PUMPIO];
|
|
|
|
|
2018-08-30 17:47:48 -04:00
|
|
|
// Supported through a connector
|
|
|
|
const DIASPORA2 = 'dspc'; // Diaspora connector
|
|
|
|
const LINKEDIN = 'lnkd'; // LinkedIn
|
2018-02-07 08:31:17 -05:00
|
|
|
const PUMPIO = 'pump'; // pump.io
|
2018-08-30 17:47:48 -04:00
|
|
|
const STATUSNET = 'stac'; // Statusnet connector
|
2018-02-07 08:31:17 -05:00
|
|
|
const TWITTER = 'twit'; // Twitter
|
2019-11-24 15:06:47 -05:00
|
|
|
const DISCOURSE = 'dscs'; // Discourse
|
2018-02-07 08:31:17 -05:00
|
|
|
|
2019-01-16 16:39:56 -05:00
|
|
|
// Dead protocols
|
|
|
|
const APPNET = 'apdn'; // app.net - Dead protocol
|
|
|
|
const FACEBOOK = 'face'; // Facebook API - Not working anymore, API is closed
|
|
|
|
const GPLUS = 'goog'; // Google+ - Dead in 2019
|
|
|
|
|
2018-08-30 17:47:48 -04:00
|
|
|
// Currently unsupported
|
|
|
|
const ICALENDAR = 'ical'; // iCalendar
|
|
|
|
const MYSPACE = 'mysp'; // MySpace
|
|
|
|
const NEWS = 'nntp'; // Network News Transfer Protocol
|
|
|
|
const PNUT = 'pnut'; // pnut.io
|
|
|
|
const XMPP = 'xmpp'; // XMPP
|
|
|
|
const ZOT = 'zot!'; // Zot!
|
2018-02-07 08:31:17 -05:00
|
|
|
|
|
|
|
const PHANTOM = 'unkn'; // Place holder
|
|
|
|
|
2021-10-02 11:44:47 -04:00
|
|
|
/**
|
|
|
|
* Returns whether the provided protocol supports following
|
|
|
|
*
|
|
|
|
* @param $protocol
|
|
|
|
* @return bool
|
|
|
|
* @throws HTTPException\InternalServerErrorException
|
|
|
|
*/
|
|
|
|
public static function supportsFollow($protocol): bool
|
|
|
|
{
|
|
|
|
if (in_array($protocol, self::NATIVE_SUPPORT)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2021-10-06 12:07:16 -04:00
|
|
|
$hook_data = [
|
|
|
|
'protocol' => $protocol,
|
|
|
|
'result' => null
|
|
|
|
];
|
|
|
|
Hook::callAll('support_follow', $hook_data);
|
2021-10-02 11:44:47 -04:00
|
|
|
|
2021-10-06 12:07:16 -04:00
|
|
|
return $hook_data['result'] === true;
|
2021-10-02 11:44:47 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns whether the provided protocol supports revoking inbound follows
|
|
|
|
*
|
|
|
|
* @param $protocol
|
|
|
|
* @return bool
|
|
|
|
* @throws HTTPException\InternalServerErrorException
|
|
|
|
*/
|
|
|
|
public static function supportsRevokeFollow($protocol): bool
|
|
|
|
{
|
|
|
|
if (in_array($protocol, self::NATIVE_SUPPORT)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2021-10-06 12:07:16 -04:00
|
|
|
$hook_data = [
|
|
|
|
'protocol' => $protocol,
|
|
|
|
'result' => null
|
|
|
|
];
|
|
|
|
Hook::callAll('support_revoke_follow', $hook_data);
|
2021-10-02 11:44:47 -04:00
|
|
|
|
2021-10-06 12:07:16 -04:00
|
|
|
return $hook_data['result'] === true;
|
2021-10-02 11:44:47 -04:00
|
|
|
}
|
|
|
|
|
2021-11-07 01:57:18 -04:00
|
|
|
/**
|
|
|
|
* Send a follow message to a remote server.
|
|
|
|
*
|
|
|
|
* @param int $uid User Id
|
|
|
|
* @param array $contact Contact being followed
|
|
|
|
* @param ?string $protocol Expected protocol
|
|
|
|
* @return bool Only returns false in the unlikely case an ActivityPub contact ID doesn't exist (???)
|
|
|
|
* @throws HTTPException\InternalServerErrorException
|
|
|
|
* @throws \ImagickException
|
|
|
|
*/
|
|
|
|
public static function follow(int $uid, array $contact, ?string $protocol = null): bool
|
|
|
|
{
|
|
|
|
$owner = User::getOwnerDataById($uid);
|
|
|
|
if (!DBA::isResult($owner)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
$protocol = $protocol ?? $contact['protocol'];
|
|
|
|
|
|
|
|
if (in_array($protocol, [Protocol::OSTATUS, Protocol::DFRN])) {
|
|
|
|
// create a follow slap
|
|
|
|
$item = [
|
|
|
|
'verb' => Activity::FOLLOW,
|
2022-09-12 17:12:11 -04:00
|
|
|
'gravity' => Item::GRAVITY_ACTIVITY,
|
2021-11-07 01:57:18 -04:00
|
|
|
'follow' => $contact['url'],
|
|
|
|
'body' => '',
|
|
|
|
'title' => '',
|
|
|
|
'guid' => '',
|
|
|
|
'uri-id' => 0,
|
|
|
|
];
|
|
|
|
|
|
|
|
$slap = OStatus::salmon($item, $owner);
|
|
|
|
|
|
|
|
if (!empty($contact['notify'])) {
|
|
|
|
Salmon::slapper($owner, $contact['notify'], $slap);
|
|
|
|
}
|
|
|
|
} elseif ($protocol == Protocol::DIASPORA) {
|
|
|
|
$contact = Diaspora::sendShare($owner, $contact);
|
|
|
|
Logger::notice('share returns: ' . $contact);
|
|
|
|
} elseif ($protocol == Protocol::ACTIVITYPUB) {
|
|
|
|
$activity_id = ActivityPub\Transmitter::activityIDFromContact($contact['id']);
|
|
|
|
if (empty($activity_id)) {
|
|
|
|
// This really should never happen
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
$success = ActivityPub\Transmitter::sendActivity('Follow', $contact['url'], $owner['uid'], $activity_id);
|
|
|
|
Logger::notice('Follow returns: ' . $success);
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2021-09-26 10:30:44 -04:00
|
|
|
/**
|
2021-10-16 21:24:34 -04:00
|
|
|
* Sends an unfollow message. Does not remove the contact
|
2021-09-26 10:30:44 -04:00
|
|
|
*
|
2021-10-16 21:24:34 -04:00
|
|
|
* @param array $contact Target public contact (uid = 0) array
|
|
|
|
* @param array $user Source local user array
|
2021-10-06 20:18:44 -04:00
|
|
|
* @return bool|null true if successful, false if not, null if no remote action was performed
|
2021-09-26 10:30:44 -04:00
|
|
|
* @throws HTTPException\InternalServerErrorException
|
|
|
|
* @throws \ImagickException
|
|
|
|
*/
|
2021-10-16 21:24:34 -04:00
|
|
|
public static function unfollow(array $contact, array $user): ?bool
|
2021-09-26 10:30:44 -04:00
|
|
|
{
|
|
|
|
if (empty($contact['network'])) {
|
|
|
|
throw new \InvalidArgumentException('Missing network key in contact array');
|
|
|
|
}
|
|
|
|
|
|
|
|
$protocol = $contact['network'];
|
|
|
|
if (($protocol == Protocol::DFRN) && !empty($contact['protocol'])) {
|
|
|
|
$protocol = $contact['protocol'];
|
|
|
|
}
|
|
|
|
|
|
|
|
if (in_array($protocol, [Protocol::OSTATUS, Protocol::DFRN])) {
|
|
|
|
// create an unfollow slap
|
2022-09-12 17:12:11 -04:00
|
|
|
$item = [
|
|
|
|
'verb' => Activity::O_UNFOLLOW,
|
|
|
|
'gravity' => Item::GRAVITY_ACTIVITY,
|
|
|
|
'follow' => $contact['url'],
|
|
|
|
'body' => '',
|
|
|
|
'title' => '',
|
|
|
|
'guid' => '',
|
|
|
|
'uri-id' => 0,
|
|
|
|
];
|
|
|
|
|
2021-09-26 10:30:44 -04:00
|
|
|
$slap = OStatus::salmon($item, $user);
|
|
|
|
|
|
|
|
if (empty($contact['notify'])) {
|
|
|
|
throw new \InvalidArgumentException('Missing expected "notify" key in OStatus/DFRN contact');
|
|
|
|
}
|
|
|
|
|
|
|
|
return Salmon::slapper($user, $contact['notify'], $slap) === 0;
|
|
|
|
} elseif ($protocol == Protocol::DIASPORA) {
|
|
|
|
return Diaspora::sendUnshare($user, $contact) > 0;
|
|
|
|
} elseif ($protocol == Protocol::ACTIVITYPUB) {
|
|
|
|
return ActivityPub\Transmitter::sendContactUndo($contact['url'], $contact['id'], $user['uid']);
|
|
|
|
}
|
|
|
|
|
2021-10-02 11:44:47 -04:00
|
|
|
// Catch-all hook for connector addons
|
2021-09-26 10:30:44 -04:00
|
|
|
$hook_data = [
|
|
|
|
'contact' => $contact,
|
2021-10-16 21:24:34 -04:00
|
|
|
'uid' => $user['uid'],
|
|
|
|
'result' => null,
|
2021-09-26 10:30:44 -04:00
|
|
|
];
|
|
|
|
Hook::callAll('unfollow', $hook_data);
|
|
|
|
|
|
|
|
return $hook_data['result'];
|
|
|
|
}
|
2021-10-02 11:44:47 -04:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Revoke an incoming follow from the provided contact
|
|
|
|
*
|
2021-10-16 21:09:49 -04:00
|
|
|
* @param array $contact Target public contact (uid == 0) array
|
|
|
|
* @param int $uid Source local user id
|
2021-10-02 15:48:20 -04:00
|
|
|
* @return bool|null true if successful, false if not, null if no action was performed
|
2021-10-02 11:44:47 -04:00
|
|
|
* @throws \Friendica\Network\HTTPException\InternalServerErrorException
|
|
|
|
* @throws \ImagickException
|
|
|
|
*/
|
2021-10-16 21:09:49 -04:00
|
|
|
public static function revokeFollow(array $contact, int $uid): ?bool
|
2021-10-02 11:44:47 -04:00
|
|
|
{
|
|
|
|
if (empty($contact['network'])) {
|
|
|
|
throw new \InvalidArgumentException('Missing network key in contact array');
|
|
|
|
}
|
|
|
|
|
|
|
|
$protocol = $contact['network'];
|
|
|
|
if ($protocol == Protocol::DFRN && !empty($contact['protocol'])) {
|
|
|
|
$protocol = $contact['protocol'];
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($protocol == Protocol::ACTIVITYPUB) {
|
2021-10-16 21:09:49 -04:00
|
|
|
return ActivityPub\Transmitter::sendContactReject($contact['url'], $contact['hub-verify'], $uid);
|
2021-10-02 11:44:47 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// Catch-all hook for connector addons
|
|
|
|
$hook_data = [
|
|
|
|
'contact' => $contact,
|
2021-10-16 21:09:49 -04:00
|
|
|
'uid' => $uid,
|
|
|
|
'result' => null,
|
2021-10-02 11:44:47 -04:00
|
|
|
];
|
|
|
|
Hook::callAll('revoke_follow', $hook_data);
|
|
|
|
|
|
|
|
return $hook_data['result'];
|
|
|
|
}
|
2021-10-02 16:00:06 -04:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Send a block message to a remote server. Only useful for connector addons.
|
|
|
|
*
|
|
|
|
* @param array $contact Public contact record to block
|
|
|
|
* @param int $uid User issuing the block
|
|
|
|
* @return bool|null true if successful, false if not, null if no action was performed
|
|
|
|
* @throws HTTPException\InternalServerErrorException
|
|
|
|
*/
|
|
|
|
public static function block(array $contact, int $uid): ?bool
|
|
|
|
{
|
|
|
|
// Catch-all hook for connector addons
|
|
|
|
$hook_data = [
|
|
|
|
'contact' => $contact,
|
|
|
|
'uid' => $uid,
|
|
|
|
'result' => null,
|
|
|
|
];
|
|
|
|
Hook::callAll('block', $hook_data);
|
|
|
|
|
|
|
|
return $hook_data['result'];
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Send an unblock message to a remote server. Only useful for connector addons.
|
|
|
|
*
|
|
|
|
* @param array $contact Public contact record to unblock
|
|
|
|
* @param int $uid User revoking the block
|
|
|
|
* @return bool|null true if successful, false if not, null if no action was performed
|
|
|
|
* @throws HTTPException\InternalServerErrorException
|
|
|
|
*/
|
|
|
|
public static function unblock(array $contact, int $uid): ?bool
|
|
|
|
{
|
|
|
|
// Catch-all hook for connector addons
|
|
|
|
$hook_data = [
|
|
|
|
'contact' => $contact,
|
|
|
|
'uid' => $uid,
|
|
|
|
'result' => null,
|
|
|
|
];
|
|
|
|
Hook::callAll('unblock', $hook_data);
|
|
|
|
|
|
|
|
return $hook_data['result'];
|
|
|
|
}
|
2018-02-07 08:31:17 -05:00
|
|
|
}
|