Merge pull request #7170 from MrPetovan/bug/6981-contact-request-blocked
Contact requests blocked
This commit is contained in:
commit
5c2cca432f
7 changed files with 80 additions and 58 deletions
|
@ -361,7 +361,7 @@ function api_call(App $a)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Logger::warning(API_LOG_PREFIX . 'not implemented', ['module' => 'api', 'action' => 'call']);
|
Logger::warning(API_LOG_PREFIX . 'not implemented', ['module' => 'api', 'action' => 'call', 'query' => $a->query_string]);
|
||||||
throw new NotImplementedException();
|
throw new NotImplementedException();
|
||||||
} catch (HTTPException $e) {
|
} catch (HTTPException $e) {
|
||||||
header("HTTP/1.1 {$e->getCode()} {$e->httpdesc}");
|
header("HTTP/1.1 {$e->getCode()} {$e->httpdesc}");
|
||||||
|
|
|
@ -2120,17 +2120,33 @@ class Contact extends BaseObject
|
||||||
return $contact;
|
return $contact;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static function addRelationship($importer, $contact, $datarray, $item = '', $sharing = false, $note = '') {
|
/**
|
||||||
|
* @param array $importer Owner (local user) data
|
||||||
|
* @param array $contact Existing owner-specific contact data we want to expand the relationship with. Optional.
|
||||||
|
* @param array $datarray An item-like array with at least the 'author-id' and 'author-url' keys for the contact. Mandatory.
|
||||||
|
* @param bool $sharing True: Contact is now sharing with Owner; False: Contact is now following Owner (default)
|
||||||
|
* @param string $note Introduction additional message
|
||||||
|
* @return bool|null True: follow request is accepted; False: relationship is rejected; Null: relationship is pending
|
||||||
|
* @throws \Friendica\Network\HTTPException\InternalServerErrorException
|
||||||
|
* @throws \ImagickException
|
||||||
|
*/
|
||||||
|
public static function addRelationship(array $importer, array $contact, array $datarray, $sharing = false, $note = '')
|
||||||
|
{
|
||||||
// Should always be set
|
// Should always be set
|
||||||
if (empty($datarray['author-id'])) {
|
if (empty($datarray['author-id'])) {
|
||||||
return;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
$fields = ['url', 'name', 'nick', 'photo', 'network'];
|
$fields = ['url', 'name', 'nick', 'photo', 'network', 'blocked'];
|
||||||
$pub_contact = DBA::selectFirst('contact', $fields, ['id' => $datarray['author-id']]);
|
$pub_contact = DBA::selectFirst('contact', $fields, ['id' => $datarray['author-id']]);
|
||||||
if (!DBA::isResult($pub_contact)) {
|
if (!DBA::isResult($pub_contact)) {
|
||||||
// Should never happen
|
// Should never happen
|
||||||
return;
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Contact is blocked at node-level
|
||||||
|
if (self::isBlocked($datarray['author-id'])) {
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
$url = defaults($datarray, 'author-link', $pub_contact['url']);
|
$url = defaults($datarray, 'author-link', $pub_contact['url']);
|
||||||
|
@ -2139,44 +2155,45 @@ class Contact extends BaseObject
|
||||||
$nick = $pub_contact['nick'];
|
$nick = $pub_contact['nick'];
|
||||||
$network = $pub_contact['network'];
|
$network = $pub_contact['network'];
|
||||||
|
|
||||||
if (is_array($contact)) {
|
if (!empty($contact)) {
|
||||||
|
// Contact is blocked at user-level
|
||||||
|
if (self::isBlockedByUser($contact['id'], $importer['id'])) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
// Make sure that the existing contact isn't archived
|
// Make sure that the existing contact isn't archived
|
||||||
self::unmarkForArchival($contact);
|
self::unmarkForArchival($contact);
|
||||||
|
|
||||||
$protocol = self::getProtocol($url, $contact['network']);
|
|
||||||
|
|
||||||
if (($contact['rel'] == self::SHARING)
|
if (($contact['rel'] == self::SHARING)
|
||||||
|| ($sharing && $contact['rel'] == self::FOLLOWER)) {
|
|| ($sharing && $contact['rel'] == self::FOLLOWER)) {
|
||||||
DBA::update('contact', ['rel' => self::FRIEND, 'writable' => true, 'pending' => false],
|
DBA::update('contact', ['rel' => self::FRIEND, 'writable' => true, 'pending' => false],
|
||||||
['id' => $contact['id'], 'uid' => $importer['uid']]);
|
['id' => $contact['id'], 'uid' => $importer['uid']]);
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($protocol == Protocol::ACTIVITYPUB) {
|
return true;
|
||||||
ActivityPub\Transmitter::sendContactAccept($contact['url'], $contact['hub-verify'], $importer['uid']);
|
|
||||||
}
|
|
||||||
|
|
||||||
// send email notification to owner?
|
|
||||||
} else {
|
} else {
|
||||||
$protocol = self::getProtocol($url, $network);
|
// send email notification to owner?
|
||||||
|
|
||||||
if (DBA::exists('contact', ['nurl' => Strings::normaliseLink($url), 'uid' => $importer['uid'], 'pending' => true])) {
|
if (DBA::exists('contact', ['nurl' => Strings::normaliseLink($url), 'uid' => $importer['uid'], 'pending' => true])) {
|
||||||
Logger::log('ignoring duplicated connection request from pending contact ' . $url);
|
Logger::log('ignoring duplicated connection request from pending contact ' . $url);
|
||||||
return;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
// create contact record
|
// create contact record
|
||||||
q("INSERT INTO `contact` (`uid`, `created`, `url`, `nurl`, `name`, `nick`, `photo`, `network`, `rel`,
|
DBA::insert('contact', [
|
||||||
`blocked`, `readonly`, `pending`, `writable`)
|
'uid' => $importer['uid'],
|
||||||
VALUES (%d, '%s', '%s', '%s', '%s', '%s', '%s', '%s', %d, 0, 0, 1, 1)",
|
'created' => DateTimeFormat::utcNow(),
|
||||||
intval($importer['uid']),
|
'url' => $url,
|
||||||
DBA::escape(DateTimeFormat::utcNow()),
|
'nurl' => Strings::normaliseLink($url),
|
||||||
DBA::escape($url),
|
'name' => $name,
|
||||||
DBA::escape(Strings::normaliseLink($url)),
|
'nick' => $nick,
|
||||||
DBA::escape($name),
|
'photo' => $photo,
|
||||||
DBA::escape($nick),
|
'network' => $network,
|
||||||
DBA::escape($photo),
|
'rel' => self::FOLLOWER,
|
||||||
DBA::escape($network),
|
'blocked' => 0,
|
||||||
intval(self::FOLLOWER)
|
'readonly' => 0,
|
||||||
);
|
'pending' => 1,
|
||||||
|
'writable' => 1,
|
||||||
|
]);
|
||||||
|
|
||||||
$contact_record = [
|
$contact_record = [
|
||||||
'id' => DBA::lastInsertId(),
|
'id' => DBA::lastInsertId(),
|
||||||
|
@ -2220,20 +2237,16 @@ class Contact extends BaseObject
|
||||||
'verb' => ($sharing ? ACTIVITY_FRIEND : ACTIVITY_FOLLOW),
|
'verb' => ($sharing ? ACTIVITY_FRIEND : ACTIVITY_FOLLOW),
|
||||||
'otype' => 'intro'
|
'otype' => 'intro'
|
||||||
]);
|
]);
|
||||||
|
|
||||||
}
|
}
|
||||||
} elseif (DBA::isResult($user) && in_array($user['page-flags'], [User::PAGE_FLAGS_SOAPBOX, User::PAGE_FLAGS_FREELOVE, User::PAGE_FLAGS_COMMUNITY])) {
|
} elseif (DBA::isResult($user) && in_array($user['page-flags'], [User::PAGE_FLAGS_SOAPBOX, User::PAGE_FLAGS_FREELOVE, User::PAGE_FLAGS_COMMUNITY])) {
|
||||||
$condition = ['uid' => $importer['uid'], 'url' => $url, 'pending' => true];
|
$condition = ['uid' => $importer['uid'], 'url' => $url, 'pending' => true];
|
||||||
DBA::update('contact', ['pending' => false], $condition);
|
DBA::update('contact', ['pending' => false], $condition);
|
||||||
|
|
||||||
$contact = DBA::selectFirst('contact', ['url', 'network', 'hub-verify'], ['id' => $contact_record['id']]);
|
return true;
|
||||||
$protocol = self::getProtocol($contact['url'], $contact['network']);
|
|
||||||
|
|
||||||
if ($protocol == Protocol::ACTIVITYPUB) {
|
|
||||||
ActivityPub\Transmitter::sendContactAccept($contact['url'], $contact['hub-verify'], $importer['uid']);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static function removeFollower($importer, $contact, array $datarray = [], $item = "")
|
public static function removeFollower($importer, $contact, array $datarray = [], $item = "")
|
||||||
|
|
|
@ -530,7 +530,7 @@ class Processor
|
||||||
DBA::update('contact', ['hub-verify' => $activity['id'], 'protocol' => Protocol::ACTIVITYPUB], ['id' => $cid]);
|
DBA::update('contact', ['hub-verify' => $activity['id'], 'protocol' => Protocol::ACTIVITYPUB], ['id' => $cid]);
|
||||||
$contact = DBA::selectFirst('contact', [], ['id' => $cid, 'network' => Protocol::NATIVE_SUPPORT]);
|
$contact = DBA::selectFirst('contact', [], ['id' => $cid, 'network' => Protocol::NATIVE_SUPPORT]);
|
||||||
} else {
|
} else {
|
||||||
$contact = false;
|
$contact = [];
|
||||||
}
|
}
|
||||||
|
|
||||||
$item = ['author-id' => Contact::getIdForURL($activity['actor']),
|
$item = ['author-id' => Contact::getIdForURL($activity['actor']),
|
||||||
|
@ -541,7 +541,11 @@ class Processor
|
||||||
// Ensure that the contact has got the right network type
|
// Ensure that the contact has got the right network type
|
||||||
self::switchContact($item['author-id']);
|
self::switchContact($item['author-id']);
|
||||||
|
|
||||||
Contact::addRelationship($owner, $contact, $item, '', false, $note);
|
$result = Contact::addRelationship($owner, $contact, $item, false, $note);
|
||||||
|
if ($result === true) {
|
||||||
|
ActivityPub\Transmitter::sendContactAccept($item['author-link'], $item['author-id'], $owner['uid']);
|
||||||
|
}
|
||||||
|
|
||||||
$cid = Contact::getIdForURL($activity['actor'], $uid);
|
$cid = Contact::getIdForURL($activity['actor'], $uid);
|
||||||
if (empty($cid)) {
|
if (empty($cid)) {
|
||||||
return;
|
return;
|
||||||
|
|
|
@ -1538,13 +1538,16 @@ class Transmitter
|
||||||
'id' => System::baseUrl() . '/activity/' . System::createGUID(),
|
'id' => System::baseUrl() . '/activity/' . System::createGUID(),
|
||||||
'type' => 'Accept',
|
'type' => 'Accept',
|
||||||
'actor' => $owner['url'],
|
'actor' => $owner['url'],
|
||||||
'object' => ['id' => $id, 'type' => 'Follow',
|
'object' => [
|
||||||
|
'id' => (string)$id,
|
||||||
|
'type' => 'Follow',
|
||||||
'actor' => $profile['url'],
|
'actor' => $profile['url'],
|
||||||
'object' => $owner['url']],
|
'object' => $owner['url']
|
||||||
|
],
|
||||||
'instrument' => self::getService(),
|
'instrument' => self::getService(),
|
||||||
'to' => [$profile['url']]];
|
'to' => [$profile['url']]];
|
||||||
|
|
||||||
Logger::log('Sending accept to ' . $target . ' for user ' . $uid . ' with id ' . $id, Logger::DEBUG);
|
Logger::debug('Sending accept to ' . $target . ' for user ' . $uid . ' with id ' . $id);
|
||||||
|
|
||||||
$signed = LDSignature::sign($data, $owner);
|
$signed = LDSignature::sign($data, $owner);
|
||||||
HTTPSignature::transmit($signed, $profile['inbox'], $uid);
|
HTTPSignature::transmit($signed, $profile['inbox'], $uid);
|
||||||
|
@ -1568,13 +1571,16 @@ class Transmitter
|
||||||
'id' => System::baseUrl() . '/activity/' . System::createGUID(),
|
'id' => System::baseUrl() . '/activity/' . System::createGUID(),
|
||||||
'type' => 'Reject',
|
'type' => 'Reject',
|
||||||
'actor' => $owner['url'],
|
'actor' => $owner['url'],
|
||||||
'object' => ['id' => $id, 'type' => 'Follow',
|
'object' => [
|
||||||
|
'id' => (string)$id,
|
||||||
|
'type' => 'Follow',
|
||||||
'actor' => $profile['url'],
|
'actor' => $profile['url'],
|
||||||
'object' => $owner['url']],
|
'object' => $owner['url']
|
||||||
|
],
|
||||||
'instrument' => self::getService(),
|
'instrument' => self::getService(),
|
||||||
'to' => [$profile['url']]];
|
'to' => [$profile['url']]];
|
||||||
|
|
||||||
Logger::log('Sending reject to ' . $target . ' for user ' . $uid . ' with id ' . $id, Logger::DEBUG);
|
Logger::debug('Sending reject to ' . $target . ' for user ' . $uid . ' with id ' . $id);
|
||||||
|
|
||||||
$signed = LDSignature::sign($data, $owner);
|
$signed = LDSignature::sign($data, $owner);
|
||||||
HTTPSignature::transmit($signed, $profile['inbox'], $uid);
|
HTTPSignature::transmit($signed, $profile['inbox'], $uid);
|
||||||
|
|
|
@ -2247,13 +2247,12 @@ class DFRN
|
||||||
// The functions below are partly used by ostatus.php as well - where we have this variable
|
// The functions below are partly used by ostatus.php as well - where we have this variable
|
||||||
$r = q("SELECT * FROM `contact` WHERE `id` = %d", intval($importer["id"]));
|
$r = q("SELECT * FROM `contact` WHERE `id` = %d", intval($importer["id"]));
|
||||||
$contact = $r[0];
|
$contact = $r[0];
|
||||||
$nickname = $contact["nick"];
|
|
||||||
|
|
||||||
// Big question: Do we need these functions? They were part of the "consume_feed" function.
|
// Big question: Do we need these functions? They were part of the "consume_feed" function.
|
||||||
// This function once was responsible for DFRN and OStatus.
|
// This function once was responsible for DFRN and OStatus.
|
||||||
if (activity_match($item["verb"], ACTIVITY_FOLLOW)) {
|
if (activity_match($item["verb"], ACTIVITY_FOLLOW)) {
|
||||||
Logger::log("New follower");
|
Logger::log("New follower");
|
||||||
Contact::addRelationship($importer, $contact, $item, $nickname);
|
Contact::addRelationship($importer, $contact, $item);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
if (activity_match($item["verb"], ACTIVITY_UNFOLLOW)) {
|
if (activity_match($item["verb"], ACTIVITY_UNFOLLOW)) {
|
||||||
|
@ -2263,7 +2262,7 @@ class DFRN
|
||||||
}
|
}
|
||||||
if (activity_match($item["verb"], ACTIVITY_REQ_FRIEND)) {
|
if (activity_match($item["verb"], ACTIVITY_REQ_FRIEND)) {
|
||||||
Logger::log("New friend request");
|
Logger::log("New friend request");
|
||||||
Contact::addRelationship($importer, $contact, $item, $nickname, true);
|
Contact::addRelationship($importer, $contact, $item, true);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
if (activity_match($item["verb"], ACTIVITY_UNFRIEND)) {
|
if (activity_match($item["verb"], ACTIVITY_UNFRIEND)) {
|
||||||
|
|
|
@ -417,13 +417,6 @@ class OStatus
|
||||||
$author = self::fetchAuthor($xpath, $entry, $importer, $contact, $stored);
|
$author = self::fetchAuthor($xpath, $entry, $importer, $contact, $stored);
|
||||||
}
|
}
|
||||||
|
|
||||||
$value = XML::getFirstNodeValue($xpath, 'atom:author/poco:preferredUsername/text()', $entry);
|
|
||||||
if ($value != "") {
|
|
||||||
$nickname = $value;
|
|
||||||
} else {
|
|
||||||
$nickname = $author["author-name"];
|
|
||||||
}
|
|
||||||
|
|
||||||
$item = array_merge($header, $author);
|
$item = array_merge($header, $author);
|
||||||
|
|
||||||
$item["uri"] = XML::getFirstNodeValue($xpath, 'atom:id/text()', $entry);
|
$item["uri"] = XML::getFirstNodeValue($xpath, 'atom:id/text()', $entry);
|
||||||
|
@ -463,7 +456,7 @@ class OStatus
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($item["verb"] == ACTIVITY_FOLLOW) {
|
if ($item["verb"] == ACTIVITY_FOLLOW) {
|
||||||
Contact::addRelationship($importer, $contact, $item, $nickname);
|
Contact::addRelationship($importer, $contact, $item);
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -68,9 +68,16 @@ class JsonLD
|
||||||
}
|
}
|
||||||
catch (Exception $e) {
|
catch (Exception $e) {
|
||||||
$normalized = false;
|
$normalized = false;
|
||||||
Logger::error('normalise error');
|
$messages = [];
|
||||||
// Sooner or later we should log some details as well - but currently this leads to memory issues
|
$currentException = $e;
|
||||||
// Logger::log('normalise error:' . substr(print_r($e, true), 0, 10000), Logger::DEBUG);
|
do {
|
||||||
|
$messages[] = $currentException->getMessage();
|
||||||
|
} while($currentException = $currentException->getPrevious());
|
||||||
|
|
||||||
|
Logger::warning('JsonLD normalize error');
|
||||||
|
Logger::notice('JsonLD normalize error', ['messages' => $messages]);
|
||||||
|
Logger::info('JsonLD normalize error', ['trace' => $e->getTraceAsString()]);
|
||||||
|
Logger::debug('JsonLD normalize error', ['jsonobj' => $jsonobj]);
|
||||||
}
|
}
|
||||||
|
|
||||||
return $normalized;
|
return $normalized;
|
||||||
|
|
Loading…
Reference in a new issue