mirror of
https://github.com/friendica/friendica
synced 2025-01-10 18:04:42 +00:00
Fix follow/unfollow
This commit is contained in:
parent
26f8392754
commit
534db0d09a
23 changed files with 236 additions and 90 deletions
|
@ -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']);
|
||||||
|
|
|
@ -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
|
||||||
|
|
|
@ -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'])) {
|
||||||
|
if ($delete) {
|
||||||
self::remove($contact['id']);
|
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])) {
|
||||||
|
if ($delete) {
|
||||||
self::remove($contact['id']);
|
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']]);
|
||||||
}
|
}
|
||||||
|
|
|
@ -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;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -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
|
||||||
*
|
*
|
||||||
|
|
|
@ -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);
|
||||||
|
|
|
@ -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']);
|
||||||
}
|
}
|
||||||
|
|
|
@ -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());
|
||||||
}
|
}
|
||||||
|
|
|
@ -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);
|
||||||
|
|
||||||
|
|
|
@ -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;
|
||||||
|
|
||||||
|
|
|
@ -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];
|
||||||
|
|
||||||
|
|
|
@ -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', ''));
|
||||||
|
|
|
@ -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]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -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 {
|
||||||
|
|
|
@ -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;
|
||||||
|
|
|
@ -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();
|
||||||
|
|
|
@ -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.'));
|
||||||
|
|
|
@ -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.'));
|
||||||
|
|
|
@ -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
|
||||||
|
|
|
@ -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);
|
||||||
|
|
|
@ -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();
|
||||||
|
|
||||||
|
|
|
@ -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);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -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);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue