use Friendica\Database\Database;
use Friendica\Database\DBA;
use Friendica\DI;
+use Friendica\Protocol\ActivityPub;
use Friendica\Util\DateTimeFormat;
use Friendica\Util\Strings;
const BTO = 12;
const BCC = 13;
+ const ACCOUNT = 1;
+ const GENERAL_COLLECTION = 2;
+ const FOLLOWER_COLLECTION = 3;
+ const PUBLIC_COLLECTION = 4;
+
const TAG_CHARACTER = [
self::HASHTAG => '#',
self::MENTION => '@',
- self::IMPLICIT_MENTION => '%',
self::EXCLUSIVE_MENTION => '!',
+ self::IMPLICIT_MENTION => '%',
];
/**
* @param integer $type
* @param string $name
* @param string $url
+ * @param integer $target
*/
- public static function store(int $uriid, int $type, string $name, string $url = '')
+ public static function store(int $uriid, int $type, string $name, string $url = '', int $target = null)
{
if ($type == self::HASHTAG) {
// Trim Unicode non-word characters
Logger::debug('Got id for contact', ['cid' => $cid, 'url' => $url]);
if (empty($cid)) {
- // The contact wasn't found in the system (most likely some dead account)
- // We ensure that we only store a single entry by overwriting the previous name
- Logger::info('URL is not a known contact, updating tag', ['url' => $url, 'name' => $name]);
- if (!DBA::exists('tag', ['name' => substr($name, 0, 96), 'url' => $url])) {
- DBA::update('tag', ['name' => substr($name, 0, 96)], ['url' => $url]);
+ $tag = DBA::selectFirst('tag', ['name', 'type'], ['url' => $url]);
+ if (!empty($tag)) {
+ if ($tag['name'] != substr($name, 0, 96)) {
+ DBA::update('tag', ['name' => substr($name, 0, 96)], ['url' => $url]);
+ }
+ if (!empty($target) && ($tag['type'] != $target)) {
+ DBA::update('tag', ['type' => $target], ['url' => $url]);
+ }
}
}
}
}
}
- $tagid = self::getID($name, $url);
+ $tagid = self::getID($name, $url, $target);
if (empty($tagid)) {
return;
}
Logger::info('Stored tag/mention', ['uri-id' => $uriid, 'tag-id' => $tagid, 'contact-id' => $cid, 'name' => $name, 'type' => $type, 'callstack' => System::callstack(8)]);
}
+ /**
+ * Fetch the target type for the given url
+ *
+ * @param string $url
+ * @param bool $fetch Fetch information via network operations
+ * @return null|int
+ */
+ public static function getTargetType(string $url, bool $fetch = true)
+ {
+ $target = null;
+
+ if (empty($url)) {
+ return $target;
+ }
+
+ $tag = DBA::selectFirst('tag', ['url', 'type'], ['url' => $url]);
+ if (!empty($tag['type'])) {
+ $target = $tag['type'];
+ if ($target != self::GENERAL_COLLECTION) {
+ Logger::debug('Found existing type', ['type' => $tag['type'], 'url' => $url]);
+ return $target;
+ }
+ }
+
+ if ($url == ActivityPub::PUBLIC_COLLECTION) {
+ $target = self::PUBLIC_COLLECTION;
+ Logger::debug('Public collection', ['url' => $url]);
+ } else {
+ if (DBA::exists('apcontact', ['followers' => $url])) {
+ $target = self::FOLLOWER_COLLECTION;
+ Logger::debug('Found collection via existing apcontact', ['url' => $url]);
+ } elseif (Contact::getIdForURL($url, 0, $fetch ? null : false)) {
+ $target = self::ACCOUNT;
+ Logger::debug('URL is an account', ['url' => $url]);
+ } elseif ($fetch && ($target != self::GENERAL_COLLECTION)) {
+ $content = ActivityPub::fetchContent($url);
+ if (!empty($content['type']) && ($content['type'] == 'OrderedCollection')) {
+ $target = self::GENERAL_COLLECTION;
+ Logger::debug('URL is an ordered collection', ['url' => $url]);
+ }
+ }
+ }
+
+ if (!empty($target) && !empty($tag['url']) && ($tag['type'] != $target)) {
+ DBA::update('tag', ['type' => $target], ['url' => $url]);
+ }
+
+ if (empty($target)) {
+ Logger::debug('No type could be detected', ['url' => $url]);
+ }
+
+ return $target;
+ }
+
/**
* Get a tag id for a given tag name and url
*
* @param string $name
* @param string $url
+ * @param int $type
* @return void
*/
- public static function getID(string $name, string $url = '')
+ public static function getID(string $name, string $url = '', int $type = null)
{
$fields = ['name' => substr($name, 0, 96), 'url' => $url];
- $tag = DBA::selectFirst('tag', ['id'], $fields);
+ $tag = DBA::selectFirst('tag', ['id', 'type'], $fields);
if (DBA::isResult($tag)) {
+ if (empty($tag['type']) && !empty($type)) {
+ DBA::update('tag', ['type' => $type], $fields);
+ }
return $tag['id'];
}
+ if (!empty($type)) {
+ $fields['type'] = $type;
+ }
+
DBA::insert('tag', $fields, Database::INSERT_IGNORE);
$tid = DBA::lastInsertId();
if (!empty($tid)) {
/**
* Get tags and mentions from the body
- *
+ *
* @param string $body Body of the post
* @param string $tags Accepted tags
*
public static function getFromBody(string $body, string $tags = null)
{
if (is_null($tags)) {
- $tags = self::TAG_CHARACTER[self::HASHTAG] . self::TAG_CHARACTER[self::MENTION] . self::TAG_CHARACTER[self::EXCLUSIVE_MENTION];
+ $tags = self::TAG_CHARACTER[self::HASHTAG] . self::TAG_CHARACTER[self::MENTION] . self::TAG_CHARACTER[self::EXCLUSIVE_MENTION];
}
if (!preg_match_all("/([" . $tags . "])\[url\=([^\[\]]*)\]([^\[\]]*)\[\/url\]/ism", $body, $result, PREG_SET_ORDER)) {
/**
* Store tags and mentions from the body
- *
+ *
* @param integer $uriid URI-Id
* @param string $body Body of the post
* @param string $tags Accepted tags
{
Logger::info('Check for tags', ['uri-id' => $uriid, 'hash' => $tags, 'callstack' => System::callstack()]);
- $result = self::getFromBody($body, $tags);
- if (empty($result)) {
- return;
+ if (is_null($tags)) {
+ $tags = self::TAG_CHARACTER[self::HASHTAG] . self::TAG_CHARACTER[self::MENTION] . self::TAG_CHARACTER[self::EXCLUSIVE_MENTION];
}
- Logger::info('Found tags', ['uri-id' => $uriid, 'hash' => $tags, 'result' => $result]);
+ // Only remove the shared data from "real" reshares
+ $shared = BBCode::fetchShareAttributes($body);
+ if (!empty($shared['guid'])) {
+ if (preg_match("/\s*\[share .*?\](.*?)\[\/share\]\s*/ism", $body, $matches)) {
+ $share_body = $matches[1];
+ }
+ $body = preg_replace("/\s*\[share .*?\].*?\[\/share\]\s*/ism", '', $body);
+ }
- foreach ($result as $tag) {
+ foreach (self::getFromBody($body, $tags) as $tag) {
self::storeByHash($uriid, $tag[1], $tag[3], $tag[2], $probing);
}
+
+ // Search for hashtags in the shared body (but only if hashtags are wanted)
+ if (!empty($share_body) && (strpos($tags, self::TAG_CHARACTER[self::HASHTAG]) !== false)) {
+ foreach (self::getFromBody($share_body, self::TAG_CHARACTER[self::HASHTAG]) as $tag) {
+ self::storeByHash($uriid, $tag[1], $tag[3], $tag[2], $probing);
+ }
+ }
}
/**
* Store raw tags (not encapsulated in links) from the body
* This function is needed in the intermediate phase.
* Later we can call item::setHashtags in advance to have all tags converted.
- *
+ *
* @param integer $uriid URI-Id
* @param string $body Body of the post
*/
*/
public static function existsForPost(int $uriid)
{
- return DBA::exists('post-tag', ['uri-id' => $uriid, 'type' => [self::HASHTAG, self::MENTION, self::IMPLICIT_MENTION, self::EXCLUSIVE_MENTION]]);
+ return DBA::exists('post-tag', ['uri-id' => $uriid, 'type' => [self::HASHTAG, self::MENTION, self::EXCLUSIVE_MENTION, self::IMPLICIT_MENTION]]);
}
/**
return;
}
- $tags = DBA::select('tag-view', ['name', 'url'], ['uri-id' => $parent_uri_id]);
+ $tags = DBA::select('tag-view', ['name', 'url'], ['uri-id' => $parent_uri_id, 'type' => [self::MENTION, self::EXCLUSIVE_MENTION, self::IMPLICIT_MENTION]]);
while ($tag = DBA::fetch($tags)) {
self::store($uri_id, self::IMPLICIT_MENTION, $tag['name'], $tag['url']);
}
* @return array
* @throws \Exception
*/
- public static function getByURIId(int $uri_id, array $type = [self::HASHTAG, self::MENTION, self::IMPLICIT_MENTION, self::EXCLUSIVE_MENTION])
+ public static function getByURIId(int $uri_id, array $type = [self::HASHTAG, self::MENTION, self::EXCLUSIVE_MENTION, self::IMPLICIT_MENTION])
{
$condition = ['uri-id' => $uri_id, 'type' => $type];
- return DBA::selectToArray('tag-view', ['type', 'name', 'url'], $condition);
+ return DBA::selectToArray('tag-view', ['type', 'name', 'url', 'tag-type'], $condition);
}
/**
* @return string tags and mentions
* @throws \Exception
*/
- public static function getCSVByURIId(int $uri_id, array $type = [self::HASHTAG, self::MENTION, self::IMPLICIT_MENTION, self::EXCLUSIVE_MENTION])
+ public static function getCSVByURIId(int $uri_id, array $type = [self::HASHTAG, self::MENTION, self::EXCLUSIVE_MENTION, self::IMPLICIT_MENTION])
{
$tag_list = [];
$tags = self::getByURIId($uri_id, $type);
/**
* Fetch the blocked tags as SQL
*
- * @return string
+ * @return string
*/
private static function getBlockedSQL()
{