You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

140 lines
4.0 KiB

<?php
namespace Telegraph;
use ORM, Exception;
use IndieWeb\MentionClient;
class Webmention {
private static $http = false;
// Returns false if the target URL is known to not accept webmentions
public static function isProbablySupported($targetURL) {
// Reject links that are known to not accept webmentions
$host = str_replace('www.','',parse_url($targetURL, PHP_URL_HOST));
if(!$host) return false;
$unsupported = [
'twitter.com',
'instagram.com',
'facebook.com',
];
if(in_array($host, $unsupported))
return false;
if(preg_match('/.+\.amazonaws\.com/', $host))
return false;
return true;
}
private static function updateStatus($webmention, $http_code, $code, $raw=null) {
$status = ORM::for_table('webmention_status')->create();
$status->webmention_id = $webmention->id;
$status->created_at = date('Y-m-d H:i:s');
if($http_code)
$status->http_code = $http_code;
$status->status = $code;
if($raw)
$status->raw_response = $raw;
$status->save();
// Post to the callback URL if one is set
if($webmention->callback) {
$payload = [
'source' => $webmention->source,
'target' => $webmention->target,
'status' => $code
];
if($webmention->webmention_endpoint) {
$payload['type'] = 'webmention';
}
if($webmention->pingback_endpoint) {
$payload['type'] = 'pingback';
}
return self::$http->post($webmention->callback, $payload);
}
}
public static function send($id, $client=false, $http=false) {
initdb();
$webmention = ORM::for_table('webmentions')->where('id', $id)->find_one();
if(!$webmention) {
echo 'Webmention '.$id.' was not found'."\n";
return;
}
if(!$client)
$client = new MentionClient();
if(!$http)
$http = new HTTP();
self::$http = $http;
// Discover the webmention or pingback endpoint
$endpoint = $client->discoverWebmentionEndpoint($webmention->target);
if(!$endpoint) {
// If no webmention endpoint found, try to send a pingback
$pingbackEndpoint = $client->discoverPingbackEndpoint($webmention->target);
// If no pingback endpoint was found, we can't do anything else
if(!$pingbackEndpoint) {
return self::updateStatus($webmention, null, 'not_supported');
}
$webmention->pingback_endpoint = $pingbackEndpoint;
$webmention->save();
$success = $client->sendPingbackToEndpoint($pingbackEndpoint, $webmention->source, $webmention->target);
return self::updateStatus($webmention, null, ($success ? 'accepted' : 'error'));
}
// There is a webmention endpoint, send the webmention now
$webmention->webmention_endpoint = $endpoint;
$webmention->save();
$params = [];
if($webmention->code) {
$params['code'] = $webmention->code;
if($webmention->realm)
$params['realm'] = $webmention->realm;
}
if($webmention->vouch) {
$params['vouch'] = $webmention->vouch;
}
$response = $client->sendWebmentionToEndpoint($endpoint, $webmention->source, $webmention->target, $params);
if(in_array($response['code'], [200,201,202])) {
$status = 'accepted';
$webmention->complete = $response['code'] == 200 ? 1 : 0;
// Check if the endpoint returned a status URL
if(array_key_exists('Location', $response['headers'])) {
$webmention->webmention_status_url = \Mf2\resolveUrl($endpoint, $response['headers']['Location']);
// TODO: queue a job to poll the endpoint for updates and deliver to the callback URL
} else {
// No status URL was returned, so we can't follow up with this later. Mark as complete.
$webmention->complete = 1;
}
} else {
$webmention->complete = 1;
$status = 'error';
}
$webmention->save();
$result = self::updateStatus($webmention, $response['code'], $status, $response['body']);
$pdo = ORM::get_db();
$pdo = null;
return $result;
}
}