mirror of
https://github.com/friendica/friendica
synced 2025-01-10 16:44:42 +00:00
Merge pull request #14151 from annando/local-link
Probing for Zot improved and Pumpio removed
This commit is contained in:
commit
c604477cac
3 changed files with 83 additions and 271 deletions
|
@ -184,6 +184,12 @@ class Media
|
||||||
{
|
{
|
||||||
if (Network::isLocalLink($media['url'])) {
|
if (Network::isLocalLink($media['url'])) {
|
||||||
$media = self::fetchLocalData($media);
|
$media = self::fetchLocalData($media);
|
||||||
|
if (preg_match('|.*?/search\?(.+)|', $media['url'], $matches)) {
|
||||||
|
return $media;
|
||||||
|
}
|
||||||
|
if (empty($media['mimetype']) || empty($media['size'])) {
|
||||||
|
Logger::debug('Unknown local link', ['url' => $media['url']]);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Fetch the mimetype or size if missing.
|
// Fetch the mimetype or size if missing.
|
||||||
|
@ -391,7 +397,7 @@ class Media
|
||||||
*/
|
*/
|
||||||
private static function fetchLocalData(array $media): array
|
private static function fetchLocalData(array $media): array
|
||||||
{
|
{
|
||||||
if (preg_match('|.*?/attach/(\d+)|', $media['url'] ?? '', $matches)) {
|
if (preg_match('|.*?/attach/(\d+)|', $media['url'], $matches)) {
|
||||||
$attachment = Attach::selectFirst(['filename', 'filetype', 'filesize'], ['id' => $matches[1]]);
|
$attachment = Attach::selectFirst(['filename', 'filetype', 'filesize'], ['id' => $matches[1]]);
|
||||||
if (!empty($attachment)) {
|
if (!empty($attachment)) {
|
||||||
$media['name'] = $attachment['filename'];
|
$media['name'] = $attachment['filename'];
|
||||||
|
@ -401,7 +407,7 @@ class Media
|
||||||
return $media;
|
return $media;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!preg_match('|.*?/photo/(.*[a-fA-F0-9])\-(.*[0-9])\..*[\w]|', $media['url'] ?? '', $matches)) {
|
if (!preg_match('|.*?/photo/(.*[a-fA-F0-9])\-(.*[0-9])\..*[\w]|', $media['url'], $matches)) {
|
||||||
return $media;
|
return $media;
|
||||||
}
|
}
|
||||||
$photo = Photo::selectFirst([], ['resource-id' => $matches[1], 'scale' => $matches[2]]);
|
$photo = Photo::selectFirst([], ['resource-id' => $matches[1], 'scale' => $matches[2]]);
|
||||||
|
|
|
@ -112,12 +112,14 @@ class Probe
|
||||||
*/
|
*/
|
||||||
private static function rearrangeData(array $data): array
|
private static function rearrangeData(array $data): array
|
||||||
{
|
{
|
||||||
$fields = ['name', 'given_name', 'family_name', 'nick', 'guid', 'url', 'addr', 'alias',
|
$fields = [
|
||||||
|
'name', 'given_name', 'family_name', 'nick', 'guid', 'url', 'addr', 'alias',
|
||||||
'photo', 'photo_medium', 'photo_small', 'header',
|
'photo', 'photo_medium', 'photo_small', 'header',
|
||||||
'account-type', 'community', 'keywords', 'location', 'about', 'xmpp', 'matrix',
|
'account-type', 'community', 'keywords', 'location', 'about', 'xmpp', 'matrix',
|
||||||
'hide', 'batch', 'notify', 'poll', 'request', 'confirm', 'subscribe', 'poco',
|
'hide', 'batch', 'notify', 'poll', 'request', 'confirm', 'subscribe', 'poco',
|
||||||
'following', 'followers', 'inbox', 'outbox', 'sharedinbox',
|
'following', 'followers', 'inbox', 'outbox', 'sharedinbox',
|
||||||
'priority', 'network', 'pubkey', 'manually-approve', 'baseurl', 'gsid'];
|
'priority', 'network', 'pubkey', 'manually-approve', 'baseurl', 'gsid'
|
||||||
|
];
|
||||||
|
|
||||||
$numeric_fields = ['gsid', 'account-type'];
|
$numeric_fields = ['gsid', 'account-type'];
|
||||||
$boolean_fields = ['hide', 'manually-approve'];
|
$boolean_fields = ['hide', 'manually-approve'];
|
||||||
|
@ -149,7 +151,7 @@ class Probe
|
||||||
}
|
}
|
||||||
|
|
||||||
$newdata['networks'] = [];
|
$newdata['networks'] = [];
|
||||||
foreach ([Protocol::DIASPORA, Protocol::OSTATUS] as $network) {
|
foreach ([Protocol::DIASPORA] as $network) {
|
||||||
if (!empty($data['networks'][$network])) {
|
if (!empty($data['networks'][$network])) {
|
||||||
$data['networks'][$network]['subscribe'] = $newdata['subscribe'] ?? '';
|
$data['networks'][$network]['subscribe'] = $newdata['subscribe'] ?? '';
|
||||||
if (empty($data['networks'][$network]['baseurl'])) {
|
if (empty($data['networks'][$network]['baseurl'])) {
|
||||||
|
@ -256,7 +258,7 @@ class Probe
|
||||||
|
|
||||||
$xml = $curlResult->getBodyString();
|
$xml = $curlResult->getBodyString();
|
||||||
$xrd = XML::parseString($xml, true);
|
$xrd = XML::parseString($xml, true);
|
||||||
$host_url = 'http://'.$host;
|
$host_url = 'http://' . $host;
|
||||||
}
|
}
|
||||||
if (!is_object($xrd)) {
|
if (!is_object($xrd)) {
|
||||||
Logger::info('No xrd object found', ['host' => $host]);
|
Logger::info('No xrd object found', ['host' => $host]);
|
||||||
|
@ -765,14 +767,9 @@ class Probe
|
||||||
}
|
}
|
||||||
if ((!$result && ($network == '')) || ($network == Protocol::OSTATUS)) {
|
if ((!$result && ($network == '')) || ($network == Protocol::OSTATUS)) {
|
||||||
$result = self::ostatus($webfinger);
|
$result = self::ostatus($webfinger);
|
||||||
} else {
|
|
||||||
$result['networks'][Protocol::OSTATUS] = self::ostatus($webfinger);
|
|
||||||
}
|
}
|
||||||
if (in_array($network, ['', Protocol::ZOT])) {
|
if (in_array($network, ['', Protocol::ZOT])) {
|
||||||
$result = self::zot($webfinger, $result, $baseurl);
|
$result = self::zot($webfinger, $result);
|
||||||
}
|
|
||||||
if ((!$result && ($network == '')) || ($network == Protocol::PUMPIO)) {
|
|
||||||
$result = self::pumpio($webfinger, $addr, $baseurl);
|
|
||||||
}
|
}
|
||||||
if (empty($result['network']) && empty($ap_profile['network']) || ($network == Protocol::FEED)) {
|
if (empty($result['network']) && empty($ap_profile['network']) || ($network == Protocol::FEED)) {
|
||||||
$result = self::feed($uri);
|
$result = self::feed($uri);
|
||||||
|
@ -812,49 +809,42 @@ class Probe
|
||||||
*
|
*
|
||||||
* @param array $webfinger Webfinger data
|
* @param array $webfinger Webfinger data
|
||||||
* @param array $data previously probed data
|
* @param array $data previously probed data
|
||||||
* @param string $baseUrl Base URL
|
|
||||||
*
|
*
|
||||||
* @return array Zot data
|
* @return array Zot data
|
||||||
* @throws HTTPException\InternalServerErrorException
|
* @throws HTTPException\InternalServerErrorException
|
||||||
*/
|
*/
|
||||||
private static function zot(array $webfinger, array $data, string $baseurl): array
|
private static function zot(array $webfinger, array $data): array
|
||||||
{
|
{
|
||||||
if (!empty($webfinger['aliases']) && is_array($webfinger['aliases'])) {
|
if (empty($webfinger['aliases']) || !is_array($webfinger['aliases'])) {
|
||||||
|
return $data;
|
||||||
|
}
|
||||||
|
|
||||||
foreach ($webfinger['aliases'] as $alias) {
|
foreach ($webfinger['aliases'] as $alias) {
|
||||||
if (substr($alias, 0, 5) == 'acct:') {
|
if (substr($alias, 0, 5) == 'acct:') {
|
||||||
$data['addr'] = substr($alias, 5);
|
$data['addr'] = substr($alias, 5);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
if (!empty($webfinger['subject']) && (substr($webfinger['subject'], 0, 5) == 'acct:')) {
|
if (!empty($webfinger['subject']) && (substr($webfinger['subject'], 0, 5) == 'acct:')) {
|
||||||
$data['addr'] = substr($webfinger['subject'], 5);
|
$data['addr'] = substr($webfinger['subject'], 5);
|
||||||
}
|
}
|
||||||
|
|
||||||
$zot_url = '';
|
|
||||||
foreach ($webfinger['links'] as $link) {
|
foreach ($webfinger['links'] as $link) {
|
||||||
if (($link['rel'] == 'http://purl.org/zot/protocol') && !empty($link['href'])) {
|
if (($link['rel'] == 'http://purl.org/zot/protocol/6.0') && !empty($link['href'])) {
|
||||||
$zot_url = $link['href'];
|
$zot_url = $link['href'];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (empty($zot_url) && !empty($data['addr']) && !empty($baseurl)) {
|
|
||||||
$condition = ['nurl' => Strings::normaliseLink($baseurl), 'platform' => ['hubzilla']];
|
|
||||||
if (!DBA::exists('gserver', $condition)) {
|
|
||||||
return $data;
|
|
||||||
}
|
|
||||||
$zot_url = $baseurl . '/.well-known/zot-info?address=' . $data['addr'];
|
|
||||||
}
|
|
||||||
|
|
||||||
if (empty($zot_url)) {
|
if (empty($zot_url)) {
|
||||||
return $data;
|
return $data;
|
||||||
}
|
}
|
||||||
|
|
||||||
$data = self::pollZot($zot_url, $data);
|
$data = self::pollZot($zot_url, $data);
|
||||||
|
|
||||||
if (!empty($data['url']) && !empty($webfinger['aliases']) && is_array($webfinger['aliases'])) {
|
if (!empty($data['url'])) {
|
||||||
foreach ($webfinger['aliases'] as $alias) {
|
foreach ($webfinger['aliases'] as $alias) {
|
||||||
if (!strstr($alias, '@') && Strings::normaliseLink($alias) != Strings::normaliseLink($data['url'])) {
|
if (Network::isValidHttpUrl($alias) && Strings::normaliseLink($alias) != Strings::normaliseLink($data['url'])) {
|
||||||
$data['alias'] = $alias;
|
$data['alias'] = $alias;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -863,9 +853,9 @@ class Probe
|
||||||
return $data;
|
return $data;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static function pollZot(string $url, array $data): array
|
private static function pollZot(string $url, array $data): array
|
||||||
{
|
{
|
||||||
$curlResult = DI::httpClient()->get($url, HttpClientAccept::JSON, [HttpClientOptions::REQUEST => HttpClientRequest::CONTACTINFO]);
|
$curlResult = DI::httpClient()->get($url, 'application/x-zot+json', [HttpClientOptions::REQUEST => HttpClientRequest::CONTACTINFO]);
|
||||||
if ($curlResult->isTimeout()) {
|
if ($curlResult->isTimeout()) {
|
||||||
return $data;
|
return $data;
|
||||||
}
|
}
|
||||||
|
@ -880,36 +870,41 @@ class Probe
|
||||||
}
|
}
|
||||||
|
|
||||||
if (empty($data['network'])) {
|
if (empty($data['network'])) {
|
||||||
if (!empty($json['protocols']) && in_array('zot', $json['protocols'])) {
|
if (!empty($json['protocols']) && in_array('zot6', $json['protocols'])) {
|
||||||
$data['network'] = Protocol::ZOT;
|
|
||||||
} elseif (!isset($json['protocols'])) {
|
|
||||||
$data['network'] = Protocol::ZOT;
|
$data['network'] = Protocol::ZOT;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!empty($json['guid']) && empty($data['guid'])) {
|
if (!empty($json['public_key'])) {
|
||||||
$data['guid'] = $json['guid'];
|
$data['pubkey'] = $json['public_key'];
|
||||||
}
|
|
||||||
if (!empty($json['key']) && empty($data['pubkey'])) {
|
|
||||||
$data['pubkey'] = $json['key'];
|
|
||||||
}
|
}
|
||||||
if (!empty($json['name'])) {
|
if (!empty($json['name'])) {
|
||||||
$data['name'] = $json['name'];
|
$data['name'] = $json['name'];
|
||||||
}
|
}
|
||||||
if (!empty($json['photo'])) {
|
if (!empty($json['username'])) {
|
||||||
$data['photo'] = $json['photo'];
|
$data['nick'] = $json['username'];
|
||||||
if (!empty($json['photo_updated'])) {
|
}
|
||||||
$data['photo'] .= '?rev=' . urlencode($json['photo_updated']);
|
if (!empty($json['photo']) && !empty($json['photo']['url'])) {
|
||||||
|
$data['photo'] = $json['photo']['url'];
|
||||||
|
}
|
||||||
|
if (!empty($json['locations'])) {
|
||||||
|
foreach ($json['locations'] as $location) {
|
||||||
|
if ($location['deleted'] || (parse_url($url, PHP_URL_HOST) != $location['host'])) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (!empty($location['address'])) {
|
||||||
|
$data['addr'] = $location['address'];
|
||||||
|
}
|
||||||
|
if (!empty($location['id_url'])) {
|
||||||
|
$data['url'] = $location['id_url'];
|
||||||
|
}
|
||||||
|
if (!empty($location['callback'])) {
|
||||||
|
$data['confirm'] = $location['callback'];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (!empty($json['address'])) {
|
|
||||||
$data['addr'] = $json['address'];
|
|
||||||
}
|
}
|
||||||
if (!empty($json['url'])) {
|
if (!empty($json['primary_location']) && !empty($json['primary_location']['connections_url'])) {
|
||||||
$data['url'] = $json['url'];
|
$data['poco'] = $json['primary_location']['connections_url'];
|
||||||
}
|
|
||||||
if (!empty($json['connections_url'])) {
|
|
||||||
$data['poco'] = $json['connections_url'];
|
|
||||||
}
|
}
|
||||||
if (isset($json['searchable'])) {
|
if (isset($json['searchable'])) {
|
||||||
$data['hide'] = !$json['searchable'];
|
$data['hide'] = !$json['searchable'];
|
||||||
|
@ -917,6 +912,8 @@ class Probe
|
||||||
if (!empty($json['public_forum'])) {
|
if (!empty($json['public_forum'])) {
|
||||||
$data['community'] = $json['public_forum'];
|
$data['community'] = $json['public_forum'];
|
||||||
$data['account-type'] = User::PAGE_FLAGS_COMMUNITY;
|
$data['account-type'] = User::PAGE_FLAGS_COMMUNITY;
|
||||||
|
} elseif ($json['channel_type'] == 'normal') {
|
||||||
|
$data['account-type'] = User::PAGE_FLAGS_NORMAL;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!empty($json['profile'])) {
|
if (!empty($json['profile'])) {
|
||||||
|
@ -958,7 +955,7 @@ class Probe
|
||||||
* @return array webfinger data
|
* @return array webfinger data
|
||||||
* @throws HTTPException\InternalServerErrorException
|
* @throws HTTPException\InternalServerErrorException
|
||||||
*/
|
*/
|
||||||
public static function webfinger(string $url, string $type): array
|
private static function webfinger(string $url, string $type): array
|
||||||
{
|
{
|
||||||
try {
|
try {
|
||||||
$curlResult = DI::httpClient()->get(
|
$curlResult = DI::httpClient()->get(
|
||||||
|
@ -1133,86 +1130,6 @@ class Probe
|
||||||
return $data;
|
return $data;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Check for valid DFRN data
|
|
||||||
*
|
|
||||||
* @param array $data DFRN data
|
|
||||||
*
|
|
||||||
* @return int Number of errors
|
|
||||||
*/
|
|
||||||
public static function validDfrn(array $data): int
|
|
||||||
{
|
|
||||||
$errors = 0;
|
|
||||||
if (!isset($data['key'])) {
|
|
||||||
$errors ++;
|
|
||||||
}
|
|
||||||
if (!isset($data['dfrn-request'])) {
|
|
||||||
$errors ++;
|
|
||||||
}
|
|
||||||
if (!isset($data['dfrn-confirm'])) {
|
|
||||||
$errors ++;
|
|
||||||
}
|
|
||||||
if (!isset($data['dfrn-notify'])) {
|
|
||||||
$errors ++;
|
|
||||||
}
|
|
||||||
if (!isset($data['dfrn-poll'])) {
|
|
||||||
$errors ++;
|
|
||||||
}
|
|
||||||
return $errors;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Fetch data from a DFRN profile page and via "noscrape"
|
|
||||||
*
|
|
||||||
* @param string $profile_link Link to the profile page
|
|
||||||
* @return array profile data
|
|
||||||
* @throws HTTPException\InternalServerErrorException
|
|
||||||
* @throws \ImagickException
|
|
||||||
*/
|
|
||||||
public static function profile(string $profile_link): array
|
|
||||||
{
|
|
||||||
$data = [];
|
|
||||||
|
|
||||||
Logger::info('Check profile', ['link' => $profile_link]);
|
|
||||||
|
|
||||||
// Fetch data via noscrape - this is faster
|
|
||||||
$noscrape_url = str_replace(['/hcard/', '/profile/'], '/noscrape/', $profile_link);
|
|
||||||
$data = self::pollNoscrape($noscrape_url, $data);
|
|
||||||
|
|
||||||
if (!isset($data['notify'])
|
|
||||||
|| !isset($data['confirm'])
|
|
||||||
|| !isset($data['request'])
|
|
||||||
|| !isset($data['poll'])
|
|
||||||
|| !isset($data['name'])
|
|
||||||
|| !isset($data['photo'])
|
|
||||||
) {
|
|
||||||
$data = self::pollHcard($profile_link, $data, true);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
if (empty($data['addr']) || empty($data['nick'])) {
|
|
||||||
$probe_data = self::uri($profile_link);
|
|
||||||
$data['addr'] = ($data['addr'] ?? '') ?: $probe_data['addr'];
|
|
||||||
$data['nick'] = ($data['nick'] ?? '') ?: $probe_data['nick'];
|
|
||||||
}
|
|
||||||
|
|
||||||
$prof_data = [
|
|
||||||
'addr' => $data['addr'],
|
|
||||||
'nick' => $data['nick'],
|
|
||||||
'dfrn-request' => $data['request'] ?? null,
|
|
||||||
'dfrn-confirm' => $data['confirm'] ?? null,
|
|
||||||
'dfrn-notify' => $data['notify'] ?? null,
|
|
||||||
'dfrn-poll' => $data['poll'] ?? null,
|
|
||||||
'photo' => $data['photo'] ?? null,
|
|
||||||
'fn' => $data['name'] ?? null,
|
|
||||||
'key' => $data['pubkey'] ?? null,
|
|
||||||
];
|
|
||||||
|
|
||||||
Logger::debug('Result', ['link' => $profile_link, 'data' => $prof_data]);
|
|
||||||
|
|
||||||
return $prof_data;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Check for DFRN contact
|
* Check for DFRN contact
|
||||||
*
|
*
|
||||||
|
@ -1276,7 +1193,8 @@ class Probe
|
||||||
$noscrape_url = str_replace('/hcard/', '/noscrape/', $hcard_url);
|
$noscrape_url = str_replace('/hcard/', '/noscrape/', $hcard_url);
|
||||||
$data = self::pollNoscrape($noscrape_url, $data);
|
$data = self::pollNoscrape($noscrape_url, $data);
|
||||||
|
|
||||||
if (isset($data['notify'])
|
if (
|
||||||
|
isset($data['notify'])
|
||||||
&& isset($data['confirm'])
|
&& isset($data['confirm'])
|
||||||
&& isset($data['request'])
|
&& isset($data['request'])
|
||||||
&& isset($data['poll'])
|
&& isset($data['poll'])
|
||||||
|
@ -1485,7 +1403,7 @@ class Probe
|
||||||
|
|
||||||
if (!empty($webfinger['aliases']) && is_array($webfinger['aliases'])) {
|
if (!empty($webfinger['aliases']) && is_array($webfinger['aliases'])) {
|
||||||
foreach ($webfinger['aliases'] as $alias) {
|
foreach ($webfinger['aliases'] as $alias) {
|
||||||
if (Strings::normaliseLink($alias) != Strings::normaliseLink($data['url']) && ! strstr($alias, '@')) {
|
if (Strings::normaliseLink($alias) != Strings::normaliseLink($data['url']) && !strstr($alias, '@')) {
|
||||||
$data['alias'] = $alias;
|
$data['alias'] = $alias;
|
||||||
} elseif (substr($alias, 0, 5) == 'acct:') {
|
} elseif (substr($alias, 0, 5) == 'acct:') {
|
||||||
$data['addr'] = substr($alias, 5);
|
$data['addr'] = substr($alias, 5);
|
||||||
|
@ -1504,7 +1422,8 @@ class Probe
|
||||||
return [];
|
return [];
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!empty($data['url'])
|
if (
|
||||||
|
!empty($data['url'])
|
||||||
&& !empty($data['guid'])
|
&& !empty($data['guid'])
|
||||||
&& !empty($data['baseurl'])
|
&& !empty($data['baseurl'])
|
||||||
&& !empty($data['pubkey'])
|
&& !empty($data['pubkey'])
|
||||||
|
@ -1549,7 +1468,8 @@ class Probe
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!empty($webfinger['subject']) && strstr($webfinger['subject'], '@')
|
if (
|
||||||
|
!empty($webfinger['subject']) && strstr($webfinger['subject'], '@')
|
||||||
&& !strstr(Strings::normaliseLink($webfinger['subject']), 'http://')
|
&& !strstr(Strings::normaliseLink($webfinger['subject']), 'http://')
|
||||||
) {
|
) {
|
||||||
$data['addr'] = str_replace('acct:', '', $webfinger['subject']);
|
$data['addr'] = str_replace('acct:', '', $webfinger['subject']);
|
||||||
|
@ -1590,13 +1510,13 @@ class Probe
|
||||||
try {
|
try {
|
||||||
$data['pubkey'] = Salmon::magicKeyToPem($pubkey);
|
$data['pubkey'] = Salmon::magicKeyToPem($pubkey);
|
||||||
} catch (\Throwable $e) {
|
} catch (\Throwable $e) {
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (isset($data['notify']) && isset($data['pubkey'])
|
if (
|
||||||
|
isset($data['notify']) && isset($data['pubkey'])
|
||||||
&& isset($data['poll'])
|
&& isset($data['poll'])
|
||||||
&& isset($data['url'])
|
&& isset($data['url'])
|
||||||
) {
|
) {
|
||||||
|
@ -1654,124 +1574,6 @@ class Probe
|
||||||
return $data;
|
return $data;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Fetch data from a pump.io profile page
|
|
||||||
*
|
|
||||||
* @param string $profile_link Link to the profile page
|
|
||||||
*
|
|
||||||
* @return array Profile data
|
|
||||||
*/
|
|
||||||
private static function pumpioProfileData(string $profile_link, string $baseurl): array
|
|
||||||
{
|
|
||||||
$curlResult = DI::httpClient()->get($profile_link, HttpClientAccept::HTML, [HttpClientOptions::REQUEST => HttpClientRequest::CONTACTINFO]);
|
|
||||||
if (!$curlResult->isSuccess() || empty($curlResult->getBodyString())) {
|
|
||||||
return [];
|
|
||||||
}
|
|
||||||
|
|
||||||
$doc = new DOMDocument();
|
|
||||||
if (!@$doc->loadHTML($curlResult->getBodyString())) {
|
|
||||||
return [];
|
|
||||||
}
|
|
||||||
|
|
||||||
$xpath = new DomXPath($doc);
|
|
||||||
|
|
||||||
$data = [];
|
|
||||||
$data['name'] = $xpath->query("//span[contains(@class, 'p-name')]")->item(0)->nodeValue;
|
|
||||||
|
|
||||||
if ($data['name'] == '') {
|
|
||||||
// This is ugly - but pump.io doesn't seem to know a better way for it
|
|
||||||
$data['name'] = trim($xpath->query("//h1[@class='media-header']")->item(0)->nodeValue);
|
|
||||||
$pos = strpos($data['name'], chr(10));
|
|
||||||
if ($pos) {
|
|
||||||
$data['name'] = trim(substr($data['name'], 0, $pos));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
$data['location'] = XML::getFirstNodeValue($xpath, "//p[contains(@class, 'p-locality')]");
|
|
||||||
|
|
||||||
if ($data['location'] == '') {
|
|
||||||
$data['location'] = XML::getFirstNodeValue($xpath, "//p[contains(@class, 'location')]");
|
|
||||||
}
|
|
||||||
|
|
||||||
$data['about'] = XML::getFirstNodeValue($xpath, "//p[contains(@class, 'p-note')]");
|
|
||||||
|
|
||||||
if ($data['about'] == '') {
|
|
||||||
$data['about'] = XML::getFirstNodeValue($xpath, "//p[contains(@class, 'summary')]");
|
|
||||||
}
|
|
||||||
|
|
||||||
$avatar = $xpath->query("//img[contains(@class, 'u-photo')]")->item(0);
|
|
||||||
if (!$avatar) {
|
|
||||||
$avatar = $xpath->query("//img[@class='img-rounded media-object']")->item(0);
|
|
||||||
}
|
|
||||||
if ($avatar) {
|
|
||||||
foreach ($avatar->attributes as $attribute) {
|
|
||||||
if (($attribute->name == 'src') && !empty($attribute->value)) {
|
|
||||||
$data['photo'] = Network::addBasePath($attribute->value, $baseurl);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return $data;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Check for pump.io contact
|
|
||||||
*
|
|
||||||
* @param array $webfinger Webfinger data
|
|
||||||
* @param string $addr
|
|
||||||
*
|
|
||||||
* @return array pump.io data
|
|
||||||
*/
|
|
||||||
private static function pumpio(array $webfinger, string $addr, string $baseurl): array
|
|
||||||
{
|
|
||||||
$data = [];
|
|
||||||
// The array is reversed to take into account the order of preference for same-rel links
|
|
||||||
// See: https://tools.ietf.org/html/rfc7033#section-4.4.4
|
|
||||||
foreach (array_reverse($webfinger['links']) as $link) {
|
|
||||||
if (($link['rel'] == 'http://webfinger.net/rel/profile-page')
|
|
||||||
&& (($link['type'] ?? '') == 'text/html')
|
|
||||||
&& ($link['href'] != '')
|
|
||||||
) {
|
|
||||||
$data['url'] = $link['href'];
|
|
||||||
} elseif (($link['rel'] == 'activity-inbox') && ($link['href'] != '')) {
|
|
||||||
$data['notify'] = $link['href'];
|
|
||||||
} elseif (($link['rel'] == 'activity-outbox') && ($link['href'] != '')) {
|
|
||||||
$data['poll'] = $link['href'];
|
|
||||||
} elseif (($link['rel'] == 'dialback') && ($link['href'] != '')) {
|
|
||||||
$data['dialback'] = $link['href'];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (isset($data['poll']) && isset($data['notify'])
|
|
||||||
&& isset($data['dialback'])
|
|
||||||
&& isset($data['url'])
|
|
||||||
) {
|
|
||||||
// by now we use these fields only for the network type detection
|
|
||||||
// So we unset all data that isn't used at the moment
|
|
||||||
unset($data['dialback']);
|
|
||||||
|
|
||||||
$data['network'] = Protocol::PUMPIO;
|
|
||||||
} else {
|
|
||||||
return [];
|
|
||||||
}
|
|
||||||
|
|
||||||
$profile_data = self::pumpioProfileData($data['url'], $baseurl);
|
|
||||||
|
|
||||||
if (!$profile_data) {
|
|
||||||
return [];
|
|
||||||
}
|
|
||||||
|
|
||||||
$data = array_merge($data, $profile_data);
|
|
||||||
|
|
||||||
if (($addr != '') && ($data['name'] != '')) {
|
|
||||||
$name = trim(str_replace($addr, '', $data['name']));
|
|
||||||
if ($name != '') {
|
|
||||||
$data['name'] = $name;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return $data;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Checks HTML page for RSS feed link
|
* Checks HTML page for RSS feed link
|
||||||
*
|
*
|
||||||
|
@ -1881,7 +1683,7 @@ class Probe
|
||||||
{
|
{
|
||||||
try {
|
try {
|
||||||
$curlResult = DI::httpClient()->get($url, HttpClientAccept::FEED_XML, [HttpClientOptions::REQUEST => HttpClientRequest::CONTACTINFO]);
|
$curlResult = DI::httpClient()->get($url, HttpClientAccept::FEED_XML, [HttpClientOptions::REQUEST => HttpClientRequest::CONTACTINFO]);
|
||||||
} catch(\Throwable $e) {
|
} catch (\Throwable $e) {
|
||||||
DI::logger()->info('Error requesting feed URL', ['url' => $url, 'exception' => $e]);
|
DI::logger()->info('Error requesting feed URL', ['url' => $url, 'exception' => $e]);
|
||||||
return [];
|
return [];
|
||||||
}
|
}
|
||||||
|
@ -2055,7 +1857,7 @@ class Probe
|
||||||
$query = isset($parts['query']) ? '?' . $parts['query'] : '';
|
$query = isset($parts['query']) ? '?' . $parts['query'] : '';
|
||||||
$fragment = isset($parts['fragment']) ? '#' . $parts['fragment'] : '';
|
$fragment = isset($parts['fragment']) ? '#' . $parts['fragment'] : '';
|
||||||
|
|
||||||
$fixed = $scheme.$host.$port.$path.$query.$fragment;
|
$fixed = $scheme . $host . $port . $path . $query . $fragment;
|
||||||
|
|
||||||
Logger::debug('Avatar fixed', ['base' => $base, 'avatar' => $avatar, 'fixed' => $fixed]);
|
Logger::debug('Avatar fixed', ['base' => $base, 'avatar' => $avatar, 'fixed' => $fixed]);
|
||||||
|
|
||||||
|
@ -2108,8 +1910,10 @@ class Probe
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check the 'noscrape' endpoint when it is a Friendica server
|
// Check the 'noscrape' endpoint when it is a Friendica server
|
||||||
$gserver = DBA::selectFirst('gserver', ['noscrape'], ["`nurl` = ? AND `noscrape` != ''",
|
$gserver = DBA::selectFirst('gserver', ['noscrape'], [
|
||||||
Strings::normaliseLink($data['baseurl'])]);
|
"`nurl` = ? AND `noscrape` != ''",
|
||||||
|
Strings::normaliseLink($data['baseurl'])
|
||||||
|
]);
|
||||||
if (!DBA::isResult($gserver)) {
|
if (!DBA::isResult($gserver)) {
|
||||||
return '';
|
return '';
|
||||||
}
|
}
|
||||||
|
@ -2207,7 +2011,7 @@ class Probe
|
||||||
|
|
||||||
foreach ($entries as $entry) {
|
foreach ($entries as $entry) {
|
||||||
$published_item = $xpath->query('atom:published/text()', $entry)->item(0);
|
$published_item = $xpath->query('atom:published/text()', $entry)->item(0);
|
||||||
$updated_item = $xpath->query('atom:updated/text()' , $entry)->item(0);
|
$updated_item = $xpath->query('atom:updated/text()', $entry)->item(0);
|
||||||
$published = !empty($published_item->nodeValue) ? DateTimeFormat::utc($published_item->nodeValue) : null;
|
$published = !empty($published_item->nodeValue) ? DateTimeFormat::utc($published_item->nodeValue) : null;
|
||||||
$updated = !empty($updated_item->nodeValue) ? DateTimeFormat::utc($updated_item->nodeValue) : null;
|
$updated = !empty($updated_item->nodeValue) ? DateTimeFormat::utc($updated_item->nodeValue) : null;
|
||||||
|
|
||||||
|
|
|
@ -386,11 +386,13 @@ class Feed
|
||||||
|
|
||||||
$orig_plink = $item['plink'];
|
$orig_plink = $item['plink'];
|
||||||
|
|
||||||
|
if (!$dryRun) {
|
||||||
try {
|
try {
|
||||||
$item['plink'] = DI::httpClient()->finalUrl($item['plink']);
|
$item['plink'] = DI::httpClient()->finalUrl($item['plink']);
|
||||||
} catch (TransferException $exception) {
|
} catch (TransferException $exception) {
|
||||||
Logger::notice('Item URL couldn\'t get expanded', ['url' => $item['plink'], 'exception' => $exception]);
|
Logger::notice('Item URL couldn\'t get expanded', ['url' => $item['plink'], 'exception' => $exception]);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (empty($item['title'])) {
|
if (empty($item['title'])) {
|
||||||
$item['title'] = XML::getFirstNodeValue($xpath, $atomns . ':title/text()', $entry);
|
$item['title'] = XML::getFirstNodeValue($xpath, $atomns . ':title/text()', $entry);
|
||||||
|
|
Loading…
Reference in a new issue