Fix follow/unfollow

This commit is contained in:
Michael 2024-07-31 04:37:17 +00:00
parent 26f8392754
commit 534db0d09a
23 changed files with 236 additions and 90 deletions

View file

@ -547,9 +547,9 @@ class Item
$item['private'] = $private_group ? ItemModel::PRIVATE : ItemModel::UNLISTED; $item['private'] = $private_group ? ItemModel::PRIVATE : ItemModel::UNLISTED;
if ($only_to_group) { if ($only_to_group) {
$cdata = Contact::getPublicAndUserContactID($group_contact['id'], $item['uid']); $pcid = Contact::getPublicContactId($group_contact['id'], $item['uid']);
if (!empty($cdata['user'])) { if ($pcid) {
$item['owner-id'] = $cdata['user']; $item['owner-id'] = $pcid;
unset($item['owner-link']); unset($item['owner-link']);
unset($item['owner-name']); unset($item['owner-name']);
unset($item['owner-avatar']); unset($item['owner-avatar']);

View file

@ -290,12 +290,12 @@ class Circle
throw new HTTPException\NotFoundException('Circle not found.'); throw new HTTPException\NotFoundException('Circle not found.');
} }
$cdata = Contact::getPublicAndUserContactID($cid, $circle['uid']); $ucid = Contact::getUserContactId($cid, $circle['uid']);
if (empty($cdata['user'])) { if (!$ucid) {
throw new HTTPException\NotFoundException('Invalid contact.'); throw new HTTPException\NotFoundException('Invalid contact.');
} }
return DBA::insert('group_member', ['gid' => $gid, 'contact-id' => $cdata['user']], Database::INSERT_IGNORE); return DBA::insert('group_member', ['gid' => $gid, 'contact-id' => $ucid], Database::INSERT_IGNORE);
} }
/** /**
@ -318,12 +318,12 @@ class Circle
throw new HTTPException\NotFoundException('Circle not found.'); throw new HTTPException\NotFoundException('Circle not found.');
} }
$cdata = Contact::getPublicAndUserContactID($cid, $circle['uid']); $ucid = Contact::getUserContactId($cid, $circle['uid']);
if (empty($cdata['user'])) { if (!$ucid) {
throw new HTTPException\NotFoundException('Invalid contact.'); throw new HTTPException\NotFoundException('Invalid contact.');
} }
return DBA::delete('group_member', ['gid' => $gid, 'contact-id' => $cid]); return DBA::delete('group_member', ['gid' => $gid, 'contact-id' => $ucid]);
} }
/** /**
@ -347,12 +347,12 @@ class Circle
} }
foreach ($contacts as $cid) { foreach ($contacts as $cid) {
$cdata = Contact::getPublicAndUserContactID($cid, $circle['uid']); $ucid = Contact::getUserContactId($cid, $circle['uid']);
if (empty($cdata['user'])) { if (!$ucid) {
throw new HTTPException\NotFoundException('Invalid contact.'); throw new HTTPException\NotFoundException('Invalid contact.');
} }
DBA::insert('group_member', ['gid' => $gid, 'contact-id' => $cdata['user']], Database::INSERT_IGNORE); DBA::insert('group_member', ['gid' => $gid, 'contact-id' => $ucid], Database::INSERT_IGNORE);
} }
} }
@ -379,12 +379,12 @@ class Circle
$contactIds = []; $contactIds = [];
foreach ($contacts as $cid) { foreach ($contacts as $cid) {
$cdata = Contact::getPublicAndUserContactID($cid, $circle['uid']); $ucid = Contact::getUserContactId($cid, $circle['uid']);
if (empty($cdata['user'])) { if (!$ucid) {
throw new HTTPException\NotFoundException('Invalid contact.'); throw new HTTPException\NotFoundException('Invalid contact.');
} }
$contactIds[] = $cdata['user']; $contactIds[] = $ucid;
} }
// Return status of deletion // Return status of deletion

View file

@ -444,12 +444,12 @@ class Contact
return false; return false;
} }
$cdata = self::getPublicAndUserContactID($cid, $uid); $ucid = self::getUserContactId($cid, $uid);
if (empty($cdata['user'])) { if (!$ucid) {
return false; return false;
} }
$condition = ['id' => $cdata['user'], 'rel' => [self::FOLLOWER, self::FRIEND]]; $condition = ['id' => $ucid, 'rel' => [self::FOLLOWER, self::FRIEND]];
if ($strict) { if ($strict) {
$condition = array_merge($condition, ['pending' => false, 'readonly' => false, 'blocked' => false]); $condition = array_merge($condition, ['pending' => false, 'readonly' => false, 'blocked' => false]);
} }
@ -495,12 +495,12 @@ class Contact
return false; return false;
} }
$cdata = self::getPublicAndUserContactID($cid, $uid); $ucid = self::getUserContactId($cid, $uid);
if (empty($cdata['user'])) { if (!$ucid) {
return false; return false;
} }
$condition = ['id' => $cdata['user'], 'rel' => [self::SHARING, self::FRIEND]]; $condition = ['id' => $ucid, 'rel' => [self::SHARING, self::FRIEND]];
if ($strict) { if ($strict) {
$condition = array_merge($condition, ['pending' => false, 'readonly' => false, 'blocked' => false]); $condition = array_merge($condition, ['pending' => false, 'readonly' => false, 'blocked' => false]);
} }
@ -671,6 +671,32 @@ class Contact
return ['public' => $pcid, 'user' => $ucid]; return ['public' => $pcid, 'user' => $ucid];
} }
/**
* Returns the public contact id of a provided contact id
*
* @param integer $cid
* @param integer $uid
* @return integer
*/
public static function getPublicContactId(int $cid, int $uid): int
{
$contact = DBA::selectFirst('account-user-view', ['pid'], ['id' => $cid, 'uid' => [0, $uid]]);
return $contact['pid'] ?? 0;
}
/**
* Returns the user contact id of a provided contact id
*
* @param integer $cid
* @param integer $uid
* @return integer
*/
public static function getUserContactId(int $cid, int $uid): int
{
$data = self::getPublicAndUserContactID($cid, $uid);
return $data['user'] ?? 0;
}
/** /**
* Helper function for "getPublicAndUserContactID" * Helper function for "getPublicAndUserContactID"
* *
@ -968,13 +994,13 @@ class Contact
} }
if (in_array($contact['rel'], [self::SHARING, self::FRIEND])) { if (in_array($contact['rel'], [self::SHARING, self::FRIEND])) {
$cdata = self::getPublicAndUserContactID($contact['id'], $contact['uid']); $pcid = self::getPublicContactId($contact['id'], $contact['uid']);
if (!empty($cdata['public'])) { if ($pcid) {
Worker::add(Worker::PRIORITY_HIGH, 'Contact\Unfollow', $cdata['public'], $contact['uid']); Worker::add(Worker::PRIORITY_HIGH, 'Contact\Unfollow', $pcid, $contact['uid']);
} }
} }
self::removeSharer($contact); self::removeSharer($contact, false);
} }
/** /**
@ -998,13 +1024,13 @@ class Contact
} }
if (in_array($contact['rel'], [self::FOLLOWER, self::FRIEND])) { if (in_array($contact['rel'], [self::FOLLOWER, self::FRIEND])) {
$cdata = self::getPublicAndUserContactID($contact['id'], $contact['uid']); $pcid = self::getPublicContactId($contact['id'], $contact['uid']);
if (!empty($cdata['public'])) { if ($pcid) {
Worker::add(Worker::PRIORITY_HIGH, 'Contact\RevokeFollow', $cdata['public'], $contact['uid']); Worker::add(Worker::PRIORITY_HIGH, 'Contact\RevokeFollow', $pcid, $contact['uid']);
} }
} }
self::removeFollower($contact); self::removeFollower($contact, false);
} }
/** /**
@ -1025,14 +1051,14 @@ class Contact
throw new \InvalidArgumentException('Unexpected public contact record'); throw new \InvalidArgumentException('Unexpected public contact record');
} }
$cdata = self::getPublicAndUserContactID($contact['id'], $contact['uid']); $pcid = self::getPublicContactId($contact['id'], $contact['uid']);
if (in_array($contact['rel'], [self::SHARING, self::FRIEND]) && !empty($cdata['public'])) { if (in_array($contact['rel'], [self::SHARING, self::FRIEND]) && $pcid) {
Worker::add(Worker::PRIORITY_HIGH, 'Contact\Unfollow', $cdata['public'], $contact['uid']); Worker::add(Worker::PRIORITY_HIGH, 'Contact\Unfollow', $pcid, $contact['uid']);
} }
if (in_array($contact['rel'], [self::FOLLOWER, self::FRIEND]) && !empty($cdata['public'])) { if (in_array($contact['rel'], [self::FOLLOWER, self::FRIEND]) && $pcid) {
Worker::add(Worker::PRIORITY_HIGH, 'Contact\RevokeFollow', $cdata['public'], $contact['uid']); Worker::add(Worker::PRIORITY_HIGH, 'Contact\RevokeFollow', $pcid, $contact['uid']);
} }
self::remove($contact['id']); self::remove($contact['id']);
@ -3009,6 +3035,10 @@ class Contact
*/ */
public static function getProtocol(string $url, string $network): string public static function getProtocol(string $url, string $network): string
{ {
if (self::isLocal($url)) {
return Protocol::ACTIVITYPUB;
}
if ($network != Protocol::DFRN) { if ($network != Protocol::DFRN) {
return $network; return $network;
} }
@ -3400,16 +3430,21 @@ class Contact
* Update the local relationship when a local user loses a follower * Update the local relationship when a local user loses a follower
* *
* @param array $contact User-specific contact (uid != 0) array * @param array $contact User-specific contact (uid != 0) array
* @param bool $delete Delete if set, otherwise set relation to "nothing" when contact had been a follower
* @return void * @return void
* @throws HTTPException\InternalServerErrorException * @throws HTTPException\InternalServerErrorException
* @throws \ImagickException * @throws \ImagickException
*/ */
public static function removeFollower(array $contact) public static function removeFollower(array $contact, bool $delete = true)
{ {
if (in_array($contact['rel'] ?? [], [self::FRIEND, self::SHARING])) { if (in_array($contact['rel'] ?? [], [self::FRIEND, self::SHARING])) {
self::update(['rel' => self::SHARING], ['id' => $contact['id']]); self::update(['rel' => self::SHARING], ['id' => $contact['id']]);
} elseif (!empty($contact['id'])) { } elseif (!empty($contact['id'])) {
self::remove($contact['id']); if ($delete) {
self::remove($contact['id']);
} else {
self::update(['rel' => self::NOTHING, 'pending' => false], ['id' => $contact['id']]);
}
} else { } else {
DI::logger()->info('Couldn\'t remove follower because of invalid contact array', ['contact' => $contact]); DI::logger()->info('Couldn\'t remove follower because of invalid contact array', ['contact' => $contact]);
return; return;
@ -3419,9 +3454,9 @@ class Contact
self::clearFollowerFollowingEndpointCache($contact['uid']); self::clearFollowerFollowingEndpointCache($contact['uid']);
$cdata = self::getPublicAndUserContactID($contact['id'], $contact['uid']); $pcid = self::getPublicContactId($contact['id'], $contact['uid']);
if (!empty($cdata['public'])) { if ($pcid) {
DI::notification()->deleteForUserByVerb($contact['uid'], Activity::FOLLOW, ['actor-id' => $cdata['public']]); DI::notification()->deleteForUserByVerb($contact['uid'], Activity::FOLLOW, ['actor-id' => $pcid]);
} }
} }
@ -3430,14 +3465,19 @@ class Contact
* Removes the contact for sharing-only protocols (feed and mail). * Removes the contact for sharing-only protocols (feed and mail).
* *
* @param array $contact User-specific contact (uid != 0) array * @param array $contact User-specific contact (uid != 0) array
* @param bool $delete Delete if set, otherwise set relation to "nothing" when contact had been a sharer
* @throws HTTPException\InternalServerErrorException * @throws HTTPException\InternalServerErrorException
*/ */
public static function removeSharer(array $contact) public static function removeSharer(array $contact, bool $delete = true)
{ {
self::clearFollowerFollowingEndpointCache($contact['uid']); self::clearFollowerFollowingEndpointCache($contact['uid']);
if ($contact['rel'] == self::SHARING || in_array($contact['network'], [Protocol::FEED, Protocol::MAIL])) { if (in_array($contact['rel'], [self::SHARING, self::NOTHING]) || in_array($contact['network'], [Protocol::FEED, Protocol::MAIL])) {
self::remove($contact['id']); if ($delete) {
self::remove($contact['id']);
} else {
self::update(['rel' => self::NOTHING, 'pending' => false], ['id' => $contact['id']]);
}
} else { } else {
self::update(['rel' => self::FOLLOWER, 'pending' => false], ['id' => $contact['id']]); self::update(['rel' => self::FOLLOWER, 'pending' => false], ['id' => $contact['id']]);
} }

View file

@ -556,9 +556,9 @@ class Item
} }
if (!empty($item['causer-id']) && Contact::isSharing($item['causer-id'], $item['uid'], true)) { if (!empty($item['causer-id']) && Contact::isSharing($item['causer-id'], $item['uid'], true)) {
$cdata = Contact::getPublicAndUserContactID($item['causer-id'], $item['uid']); $ucid = Contact::getUserContactId($item['causer-id'], $item['uid']);
if (!empty($cdata['user'])) { if ($ucid) {
return $cdata['user']; return $ucid;
} }
} }
@ -2632,12 +2632,12 @@ class Item
return; return;
} }
$cdata = Contact::getPublicAndUserContactID($item['author-id'], $item['uid']); $ucid = Contact::getUserContactId($item['author-id'], $item['uid']);
if (empty($cdata['user']) || ($cdata['user'] != $item['contact-id'])) { if (!$ucid || ($ucid != $item['contact-id'])) {
return; return;
} }
if (!DBA::exists('contact', ['id' => $cdata['user'], 'remote_self' => LocalRelationship::MIRROR_NATIVE_RESHARE])) { if (!DBA::exists('contact', ['id' => $ucid, 'remote_self' => LocalRelationship::MIRROR_NATIVE_RESHARE])) {
return; return;
} }

View file

@ -412,6 +412,29 @@ class User
return 0; return 0;
} }
/**
* Returns the user id of a given contact id
*
* @param int $cid
*
* @return integer user id
* @throws Exception
*/
public static function getIdForContactId(int $cid): int
{
$account = Contact::selectFirstAccountUser(['pid', 'self', 'uid'], ['id' => $cid]);
if (empty($account['pid'])) {
return 0;
}
if ($account['self']) {
return $account['uid'];
}
$self = Contact::selectFirstAccountUser(['uid'], ['pid' => $cid, 'self' => true]);
return $self['uid'] ?? 0;
}
/** /**
* Get a user based on its email * Get a user based on its email
* *

View file

@ -43,9 +43,9 @@ class Block extends BaseApi
Contact\User::setBlocked($this->parameters['id'], $uid, true); Contact\User::setBlocked($this->parameters['id'], $uid, true);
$cdata = Contact::getPublicAndUserContactID($this->parameters['id'], $uid); $ucid = Contact::getUserContactId($this->parameters['id'], $uid);
if (!empty($cdata['user'])) { if ($ucid) {
$contact = Contact::getById($cdata['user']); $contact = Contact::getById($ucid);
if (!empty($contact)) { if (!empty($contact)) {
// Mastodon-expected behavior: relationship is severed on block // Mastodon-expected behavior: relationship is severed on block
Contact::terminateFriendship($contact); Contact::terminateFriendship($contact);

View file

@ -51,9 +51,9 @@ class Lists extends BaseApi
$lists = []; $lists = [];
$cdata = Contact::getPublicAndUserContactID($id, $uid); $ucid = Contact::getUserContactId($id, $uid);
if (!empty($cdata['user'])) { if ($ucid) {
$circles = DBA::select('group_member', ['gid'], ['contact-id' => $cdata['user']]); $circles = DBA::select('group_member', ['gid'], ['contact-id' => $ucid]);
while ($circle = DBA::fetch($circles)) { while ($circle = DBA::fetch($circles)) {
$lists[] = DI::mstdnList()->createFromCircleId($circle['gid']); $lists[] = DI::mstdnList()->createFromCircleId($circle['gid']);
} }

View file

@ -45,12 +45,12 @@ class Note extends BaseApi
'comment' => '', 'comment' => '',
], $request); ], $request);
$cdata = Contact::getPublicAndUserContactID($this->parameters['id'], $uid); $ucid = Contact::getUserContactId($this->parameters['id'], $uid);
if (empty($cdata['user'])) { if (!$ucid) {
$this->logAndJsonError(404, $this->errorFactory->RecordNotFound()); $this->logAndJsonError(404, $this->errorFactory->RecordNotFound());
} }
Contact::update(['info' => $request['comment']], ['id' => $cdata['user']]); Contact::update(['info' => $request['comment']], ['id' => $ucid]);
$this->jsonExit(DI::mstdnRelationship()->createFromContactId($this->parameters['id'], $uid)->toArray()); $this->jsonExit(DI::mstdnRelationship()->createFromContactId($this->parameters['id'], $uid)->toArray());
} }

View file

@ -40,12 +40,12 @@ class Unfollow extends BaseApi
$this->logAndJsonError(422, $this->errorFactory->UnprocessableEntity()); $this->logAndJsonError(422, $this->errorFactory->UnprocessableEntity());
} }
$cdata = Contact::getPublicAndUserContactID($this->parameters['id'], $uid); $ucid = Contact::getUserContactId($this->parameters['id'], $uid);
if (empty($cdata['user'])) { if (!$ucid) {
$this->logAndJsonError(404, $this->errorFactory->RecordNotFound()); $this->logAndJsonError(404, $this->errorFactory->RecordNotFound());
} }
$contact = Contact::getById($cdata['user']); $contact = Contact::getById($ucid);
Contact::unfollow($contact); Contact::unfollow($contact);

View file

@ -21,7 +21,6 @@
namespace Friendica\Module\Api\Mastodon; namespace Friendica\Module\Api\Mastodon;
use Friendica\Core\System;
use Friendica\DI; use Friendica\DI;
use Friendica\Model\Contact; use Friendica\Model\Contact;
use Friendica\Module\BaseApi; use Friendica\Module\BaseApi;
@ -47,12 +46,12 @@ class FollowRequests extends BaseApi
$this->checkAllowedScope(self::SCOPE_FOLLOW); $this->checkAllowedScope(self::SCOPE_FOLLOW);
$uid = self::getCurrentUserID(); $uid = self::getCurrentUserID();
$cdata = Contact::getPublicAndUserContactID($this->parameters['id'], $uid); $ucid = Contact::getUserContactId($this->parameters['id'], $uid);
if (empty($cdata['user'])) { if (!$ucid) {
throw new HTTPException\NotFoundException('Contact not found'); throw new HTTPException\NotFoundException('Contact not found');
} }
$introduction = DI::intro()->selectForContact($cdata['user']); $introduction = DI::intro()->selectForContact($ucid);
$contactId = $introduction->cid; $contactId = $introduction->cid;

View file

@ -110,8 +110,7 @@ class ListTimeline extends BaseApi
private function getStatusesForGroup(int $uid, array $request): array private function getStatusesForGroup(int $uid, array $request): array
{ {
$cdata = Contact::getPublicAndUserContactID((int)substr($this->parameters['id'], 6), $uid); $cid = Contact::getPublicContactId((int)substr($this->parameters['id'], 6), $uid);
$cid = $cdata['public'];
$condition = ["(`uid` = ? OR (`uid` = ? AND NOT `global`))", 0, $uid]; $condition = ["(`uid` = ? OR (`uid` = ? AND NOT `global`))", 0, $uid];

View file

@ -81,9 +81,9 @@ class NewDM extends BaseApi
} }
} }
$cdata = Contact::getPublicAndUserContactID($cid, $uid); $ucid = Contact::getUserContactId($cid, $uid);
$id = Mail::send($uid, $cdata['user'], $request['text'], $sub, $replyto); $id = Mail::send($uid, $ucid, $request['text'], $sub, $replyto);
if ($id > -1) { if ($id > -1) {
$ret = $this->directMessage->createFromMailId($id, $uid, $this->getRequestValue($request, 'getText', '')); $ret = $this->directMessage->createFromMailId($id, $uid, $this->getRequestValue($request, 'getText', ''));

View file

@ -88,9 +88,9 @@ abstract class DirectMessagesEndpoint extends BaseApi
$cid = BaseApi::getContactIDForSearchterm($this->getRequestValue($request, 'screen_name', ''), $this->getRequestValue($request, 'profileurl', ''), $this->getRequestValue($request, 'user_id', 0), 0); $cid = BaseApi::getContactIDForSearchterm($this->getRequestValue($request, 'screen_name', ''), $this->getRequestValue($request, 'profileurl', ''), $this->getRequestValue($request, 'user_id', 0), 0);
if (!empty($cid)) { if (!empty($cid)) {
$cdata = Contact::getPublicAndUserContactID($cid, $uid); $ucid = Contact::getUserContactId($cid, $uid);
if (!empty($cdata['user'])) { if ($ucid) {
$condition = DBA::mergeConditions($condition, ["`contact-id` = ?", $cdata['user']]); $condition = DBA::mergeConditions($condition, ["`contact-id` = ?", $ucid]);
} }
} }

View file

@ -71,13 +71,13 @@ class Destroy extends ContactEndpoint
} }
// Get Contact by given id // Get Contact by given id
$cdata = Contact::getPublicAndUserContactID($contact_id, $uid); $ucid = Contact::getUserContactId($contact_id, $uid);
if (!empty($cdata['user'])) { if (!$ucid) {
Logger::notice(BaseApi::LOG_PREFIX . 'Not following contact', ['module' => 'api', 'action' => 'friendships_destroy']); Logger::notice(BaseApi::LOG_PREFIX . 'Not following contact', ['module' => 'api', 'action' => 'friendships_destroy']);
throw new HTTPException\NotFoundException('Not following Contact'); throw new HTTPException\NotFoundException('Not following Contact');
} }
$contact = Contact::getById($cdata['user']); $contact = Contact::getById($ucid);
$user = $this->twitterUser->createFromContactId($contact_id, $uid, true)->toArray(); $user = $this->twitterUser->createFromContactId($contact_id, $uid, true)->toArray();
try { try {

View file

@ -55,9 +55,9 @@ class Show extends ContactEndpoint
$following = false; $following = false;
if ($source_cid == Contact::getPublicIdByUserId($uid)) { if ($source_cid == Contact::getPublicIdByUserId($uid)) {
$cdata = Contact::getPublicAndUserContactID($target_cid, $uid); $ucid = Contact::getUserContactId($target_cid, $uid);
if (!empty($cdata['user'])) { if ($ucid) {
$usercontact = Contact::getById($cdata['user'], ['rel']); $usercontact = Contact::getById($ucid, ['rel']);
switch ($usercontact['rel'] ?? Contact::NOTHING) { switch ($usercontact['rel'] ?? Contact::NOTHING) {
case Contact::FOLLOWER: case Contact::FOLLOWER:
$follower = true; $follower = true;

View file

@ -253,7 +253,7 @@ class Contact extends BaseModule
$sql_extra = " AND `archive` AND NOT `blocked` AND NOT `pending`"; $sql_extra = " AND `archive` AND NOT `blocked` AND NOT `pending`";
break; break;
case 'pending': case 'pending':
$sql_extra = " AND `pending` AND NOT `archive` AND NOT `failed` AND ((`rel` = ?) $sql_extra = " AND `pending` AND NOT `archive` AND ((`rel` = ?)
OR `id` IN (SELECT `contact-id` FROM `intro` WHERE `intro`.`uid` = ? AND NOT `ignore`))"; OR `id` IN (SELECT `contact-id` FROM `intro` WHERE `intro`.`uid` = ? AND NOT `ignore`))";
$sql_values[] = Model\Contact::SHARING; $sql_values[] = Model\Contact::SHARING;
$sql_values[] = DI::userSession()->getLocalUserId(); $sql_values[] = DI::userSession()->getLocalUserId();

View file

@ -215,7 +215,7 @@ class Follow extends BaseModule
$this->baseUrl->redirect($returnPath); $this->baseUrl->redirect($returnPath);
} elseif (!empty($result['cid'])) { } elseif (!empty($result['cid'])) {
$this->baseUrl->redirect('contact/' . $result['cid']); $this->baseUrl->redirect('contact/' . Contact::getPublicContactId($result['cid'], $this->session->getLocalUserId()));
} }
$this->sysMessages->addNotice($this->t('The contact could not be added.')); $this->sysMessages->addNotice($this->t('The contact could not be added.'));

View file

@ -91,8 +91,8 @@ class Profile extends BaseModule
// Backward compatibility: The update still needs a user-specific contact ID // Backward compatibility: The update still needs a user-specific contact ID
// Change to user-contact table check by version 2022.03 // Change to user-contact table check by version 2022.03
$cdata = Contact::getPublicAndUserContactID($contact_id, $this->session->getLocalUserId()); $ucid = Contact::getUserContactId($contact_id, $this->session->getLocalUserId());
if (empty($cdata['user']) || !$this->db->exists('contact', ['id' => $cdata['user'], 'deleted' => false])) { if (!$ucid || !$this->db->exists('contact', ['id' => $ucid, 'deleted' => false])) {
return; return;
} }
@ -134,14 +134,14 @@ class Profile extends BaseModule
} }
if (isset($request['channel_frequency'])) { if (isset($request['channel_frequency'])) {
Contact\User::setChannelFrequency($cdata['user'], $this->session->getLocalUserId(), $request['channel_frequency']); Contact\User::setChannelFrequency($ucid, $this->session->getLocalUserId(), $request['channel_frequency']);
} }
if (isset($request['channel_only'])) { if (isset($request['channel_only'])) {
Contact\User::setChannelOnly($cdata['user'], $this->session->getLocalUserId(), $request['channel_only']); Contact\User::setChannelOnly($ucid, $this->session->getLocalUserId(), $request['channel_only']);
} }
if (!Contact::update($fields, ['id' => $cdata['user'], 'uid' => $this->session->getLocalUserId()])) { if (!Contact::update($fields, ['id' => $ucid, 'uid' => $this->session->getLocalUserId()])) {
$this->systemMessages->addNotice($this->t('Failed to update contact record.')); $this->systemMessages->addNotice($this->t('Failed to update contact record.'));
} }
} }
@ -164,6 +164,18 @@ class Profile extends BaseModule
throw new HTTPException\NotFoundException($this->t('Contact not found.')); throw new HTTPException\NotFoundException($this->t('Contact not found.'));
} }
// Fetch the protocol from the user's contact.
$usercontact = Contact::getById($data['user'], ['network', 'protocol']);
if ($this->db->isResult($usercontact)) {
$contact['network'] = $usercontact['network'];
$contact['protocol'] = $usercontact['protocol'];
}
if (empty($contact['network']) && Contact::isLocal($contact['url']) ) {
$contact['network'] = Protocol::DFRN;
$contact['protocol'] = Protocol::ACTIVITYPUB;
}
// Don't display contacts that are about to be deleted // Don't display contacts that are about to be deleted
if ($this->db->isResult($contact) && (!empty($contact['deleted']) || !empty($contact['network']) && $contact['network'] == Protocol::PHANTOM)) { if ($this->db->isResult($contact) && (!empty($contact['deleted']) || !empty($contact['network']) && $contact['network'] == Protocol::PHANTOM)) {
throw new HTTPException\NotFoundException($this->t('Contact not found.')); throw new HTTPException\NotFoundException($this->t('Contact not found.'));

View file

@ -30,6 +30,7 @@ use Friendica\Core\Renderer;
use Friendica\Database\Database; use Friendica\Database\Database;
use Friendica\DI; use Friendica\DI;
use Friendica\Model; use Friendica\Model;
use Friendica\Model\Contact as ModelContact;
use Friendica\Module\Contact; use Friendica\Module\Contact;
use Friendica\Module\Response; use Friendica\Module\Response;
use Friendica\Module\Security\Login; use Friendica\Module\Security\Login;
@ -90,7 +91,7 @@ class Revoke extends BaseModule
DI::sysmsg()->addNotice($this->t('Follow was successfully revoked.')); DI::sysmsg()->addNotice($this->t('Follow was successfully revoked.'));
$this->baseUrl->redirect('contact/' . $this->parameters['id']); $this->baseUrl->redirect('contact/' . ModelContact::getPublicContactId($this->parameters['id'], DI::userSession()->getLocalUserId()));
} }
protected function content(array $request = []): string protected function content(array $request = []): string

View file

@ -168,7 +168,7 @@ class Unfollow extends \Friendica\BaseModule
$this->baseUrl->redirect($base_return_path); $this->baseUrl->redirect($base_return_path);
} }
$return_path = $base_return_path . '/' . $contact['id']; $return_path = $base_return_path . '/' . Contact::getPublicContactId($contact['id'], $uid);
try { try {
Contact::unfollow($contact); Contact::unfollow($contact);

View file

@ -36,6 +36,7 @@ use Friendica\Network\HTTPClient\Capability\ICanHandleHttpResponses;
use Friendica\Network\HTTPClient\Client\HttpClientAccept; use Friendica\Network\HTTPClient\Client\HttpClientAccept;
use Friendica\Network\HTTPClient\Client\HttpClientOptions; use Friendica\Network\HTTPClient\Client\HttpClientOptions;
use Friendica\Network\HTTPClient\Client\HttpClientRequest; use Friendica\Network\HTTPClient\Client\HttpClientRequest;
use Friendica\Protocol\ActivityPub\Receiver;
/** /**
* Implements HTTP Signatures per draft-cavage-http-signatures-07. * Implements HTTP Signatures per draft-cavage-http-signatures-07.
@ -315,6 +316,55 @@ class HTTPSignature
return $postResult; return $postResult;
} }
/**
* Route activities locally
*
* @param array $data
* @param string $target
* @param array $owner
* @return boolean
*/
private static function routeLocal(array $data, string $target, array $owner): bool
{
$uid = self::getUserIdForInbox($target);
if (is_null($uid)) {
return false;
}
$activity = JsonLD::compact($data);
$type = JsonLD::fetchElement($activity, '@type');
$trust_source = true;
$object_data = Receiver::prepareObjectData($activity, $uid, true, $trust_source, $owner['url']);
if (empty($object_data)) {
return false;
}
Logger::debug('Process directly', ['uid' => $uid, 'target' => $target, 'type' => $type]);
return Receiver::routeActivities($object_data, $type, true, true, $uid);
}
/**
* Fetch the user id for a given inbox
*
* @param string $inbox
* @return integer|null
*/
private static function getUserIdForInbox(string $inbox): ?int
{
$gsid = GServer::getID(DI::baseUrl());
if (!$gsid) {
return null;
}
if (DBA::exists('apcontact', ['gsid' => $gsid, 'sharedinbox' => $inbox])) {
return 0;
}
$apcontact = DBA::selectFirst('apcontact', ['url'], ['gsid' => $gsid, 'inbox' => $inbox]);
if (empty($apcontact['url'])) {
return null;
}
return User::getIdForURL($apcontact['url']);
}
/** /**
* Transmit given data to a target for a user * Transmit given data to a target for a user
* *
@ -326,6 +376,10 @@ class HTTPSignature
*/ */
public static function transmit(array $data, string $target, array $owner): bool public static function transmit(array $data, string $target, array $owner): bool
{ {
if (DI::baseUrl()->isLocalUrl($target) && self::routeLocal($data, $target, $owner)) {
return true;
}
$postResult = self::post($data, $target, $owner); $postResult = self::post($data, $target, $owner);
$return_code = $postResult->getReturnCode(); $return_code = $postResult->getReturnCode();

View file

@ -42,7 +42,12 @@ class RevokeFollow
*/ */
public static function execute(int $cid, int $uid) public static function execute(int $cid, int $uid)
{ {
$contact = Contact::getById($cid); $ucid = Contact::getUserContactId($cid, $uid);
if (!$ucid) {
return;
}
$contact = Contact::getById($ucid);
if (empty($contact)) { if (empty($contact)) {
return; return;
} }
@ -53,7 +58,11 @@ class RevokeFollow
} }
if (!Protocol::revokeFollow($contact, $owner)) { if (!Protocol::revokeFollow($contact, $owner)) {
Worker::defer(self::WORKER_DEFER_LIMIT); if (!Worker::defer(self::WORKER_DEFER_LIMIT)) {
Contact::removeFollower($contact);
}
} else {
Contact::removeFollower($contact);
} }
} }
} }

View file

@ -41,7 +41,12 @@ class Unfollow
*/ */
public static function execute(int $cid, int $uid) public static function execute(int $cid, int $uid)
{ {
$contact = Contact::getById($cid); $ucid = Contact::getUserContactId($cid, $uid);
if (!$ucid) {
return;
}
$contact = Contact::getById($ucid);
if (empty($contact)) { if (empty($contact)) {
return; return;
} }
@ -53,7 +58,11 @@ class Unfollow
$result = Protocol::unfollow($contact, $owner); $result = Protocol::unfollow($contact, $owner);
if ($result === false) { if ($result === false) {
Worker::defer(self::WORKER_DEFER_LIMIT); if (!Worker::defer(self::WORKER_DEFER_LIMIT)) {
Contact::removeSharer($contact);
}
} else {
Contact::removeSharer($contact);
} }
} }
} }