2022-05-19 19:24:21 +00:00
< ? php
/**
* @ copyright Copyright ( C ) 2010 - 2022 , the Friendica project
*
* @ license GNU AGPL version 3 or any later version
*
* This program is free software : you can redistribute it and / or modify
* it under the terms of the GNU Affero General Public License as
* published by the Free Software Foundation , either version 3 of the
* License , or ( at your option ) any later version .
*
* This program is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU Affero General Public License for more details .
*
* You should have received a copy of the GNU Affero General Public License
* along with this program . If not , see < https :// www . gnu . org / licenses />.
*
*/
namespace Friendica\Protocol\ActivityPub ;
use Friendica\Core\Logger ;
2022-05-24 07:02:42 +00:00
use Friendica\Database\DBA ;
2022-05-19 19:24:21 +00:00
use Friendica\DI ;
use Friendica\Model\Contact ;
use Friendica\Model\GServer ;
use Friendica\Model\Post ;
use Friendica\Protocol\ActivityPub ;
use Friendica\Util\HTTPSignature ;
use Friendica\Worker\Delivery as WorkerDelivery ;
class Delivery
{
2022-05-24 07:02:42 +00:00
/**
* Deliver posts to the given inbox
*
* @ param string $inbox
* @ return array with the elements " success " and " uri_ids " of the failed posts
*/
2022-05-20 04:42:10 +00:00
public static function deliver ( string $inbox ) : array
2022-05-19 19:24:21 +00:00
{
$uri_ids = [];
$posts = Post\Delivery :: selectForInbox ( $inbox );
$serverfail = false ;
foreach ( $posts as $post ) {
if ( ! $serverfail ) {
$result = self :: deliverToInbox ( $post [ 'command' ], 0 , $inbox , $post [ 'uid' ], $post [ 'receivers' ], $post [ 'uri-id' ]);
if ( $result [ 'serverfailure' ]) {
// In a timeout situation we assume that every delivery to that inbox will time out.
// So we set the flag and try all deliveries at a later time.
Logger :: info ( 'Inbox delivery has a server failure' , [ 'inbox' => $inbox ]);
$serverfail = true ;
}
}
2022-05-24 07:02:42 +00:00
if ( $serverfail || ( ! $result [ 'success' ] && ! $result [ 'drop' ])) {
2022-05-19 19:24:21 +00:00
$uri_ids [] = $post [ 'uri-id' ];
}
}
Logger :: debug ( 'Inbox delivery done' , [ 'inbox' => $inbox , 'posts' => count ( $posts ), 'failed' => count ( $uri_ids ), 'serverfailure' => $serverfail ]);
return [ 'success' => empty ( $uri_ids ), 'uri_ids' => $uri_ids ];
}
2022-05-24 07:02:42 +00:00
/**
* Deliver the given post to the given inbox
*
* @ param string $cmd
* @ param integer $item_id
* @ param string $inbox
* @ param integer $uid
* @ param array $receivers
* @ param integer $uri_id
* @ return array
*/
2022-05-19 19:24:21 +00:00
public static function deliverToInbox ( string $cmd , int $item_id , string $inbox , int $uid , array $receivers , int $uri_id ) : array
{
if ( empty ( $item_id ) && ! empty ( $uri_id ) && ! empty ( $uid )) {
$item = Post :: selectFirst ([ 'id' , 'parent' , 'origin' ], [ 'uri-id' => $uri_id , 'uid' => [ $uid , 0 ]], [ 'order' => [ 'uid' => true ]]);
if ( empty ( $item [ 'id' ])) {
2022-08-30 19:45:30 +00:00
Logger :: warning ( 'Item not found, removing delivery' , [ 'uri-id' => $uri_id , 'uid' => $uid , 'cmd' => $cmd , 'inbox' => $inbox ]);
2022-05-19 19:24:21 +00:00
Post\Delivery :: remove ( $uri_id , $inbox );
return true ;
} else {
$item_id = $item [ 'id' ];
}
}
$success = true ;
$serverfail = false ;
2022-05-24 07:02:42 +00:00
$drop = false ;
2022-05-19 19:24:21 +00:00
if ( $cmd == WorkerDelivery :: MAIL ) {
$data = ActivityPub\Transmitter :: createActivityFromMail ( $item_id );
if ( ! empty ( $data )) {
$success = HTTPSignature :: transmit ( $data , $inbox , $uid );
}
} elseif ( $cmd == WorkerDelivery :: SUGGESTION ) {
$success = ActivityPub\Transmitter :: sendContactSuggestion ( $uid , $inbox , $item_id );
} elseif ( $cmd == WorkerDelivery :: RELOCATION ) {
// @todo Implementation pending
} elseif ( $cmd == WorkerDelivery :: REMOVAL ) {
$success = ActivityPub\Transmitter :: sendProfileDeletion ( $uid , $inbox );
} elseif ( $cmd == WorkerDelivery :: PROFILEUPDATE ) {
$success = ActivityPub\Transmitter :: sendProfileUpdate ( $uid , $inbox );
} else {
2022-07-29 14:17:53 +00:00
$data = ActivityPub\Transmitter :: createCachedActivityFromItem ( $item_id );
2022-05-19 19:24:21 +00:00
if ( ! empty ( $data )) {
$timestamp = microtime ( true );
$response = HTTPSignature :: post ( $data , $inbox , $uid );
$runtime = microtime ( true ) - $timestamp ;
$success = $response -> isSuccess ();
$serverfail = $response -> isTimeout ();
if ( ! $success ) {
2022-05-24 07:02:42 +00:00
// 5xx errors are problems on the server. We don't need to continue delivery then.
2022-05-19 19:24:21 +00:00
if ( ! $serverfail && ( $response -> getReturnCode () >= 500 ) && ( $response -> getReturnCode () <= 599 )) {
$serverfail = true ;
}
2022-05-24 07:02:42 +00:00
// A 404 means that the inbox doesn't exist. We can stop the delivery here.
if ( ! $serverfail && ( $response -> getReturnCode () == 404 )) {
$serverfail = true ;
}
2022-05-19 19:24:21 +00:00
$xrd_timeout = DI :: config () -> get ( 'system' , 'xrd_timeout' );
if ( ! $serverfail && $xrd_timeout && ( $runtime > $xrd_timeout )) {
$serverfail = true ;
}
2022-05-24 07:02:42 +00:00
2022-05-19 19:24:21 +00:00
$curl_timeout = DI :: config () -> get ( 'system' , 'curl_timeout' );
if ( ! $serverfail && $curl_timeout && ( $runtime > $curl_timeout )) {
$serverfail = true ;
}
2022-05-24 07:02:42 +00:00
// Resubscribe to relay server upon client error
if ( ! $serverfail && ( $response -> getReturnCode () >= 400 ) && ( $response -> getReturnCode () <= 499 )) {
$actor = self :: fetchActorForRelayInbox ( $inbox );
if ( ! empty ( $actor )) {
$drop = ! ActivityPub\Transmitter :: sendRelayFollow ( $actor );
Logger :: notice ( 'Resubscribed to relay' , [ 'url' => $actor , 'success' => ! $drop ]);
2022-05-24 12:27:35 +00:00
} elseif ( $cmd = WorkerDelivery :: DELETION ) {
// Remote systems not always accept our deletion requests, so we drop them if rejected.
// Situation is: In Friendica we allow the thread owner to delete foreign comments to their thread.
// Most AP systems don't allow this, so they will reject the deletion request.
$drop = true ;
2022-05-24 07:02:42 +00:00
}
}
Logger :: info ( 'Delivery failed' , [ 'retcode' => $response -> getReturnCode (), 'serverfailure' => $serverfail , 'drop' => $drop , 'runtime' => round ( $runtime , 3 ), 'uri-id' => $uri_id , 'uid' => $uid , 'item_id' => $item_id , 'cmd' => $cmd , 'inbox' => $inbox ]);
2022-05-19 19:24:21 +00:00
}
if ( $uri_id ) {
if ( $success ) {
Post\Delivery :: remove ( $uri_id , $inbox );
} else {
Post\Delivery :: incrementFailed ( $uri_id , $inbox );
}
}
}
}
self :: setSuccess ( $receivers , $success );
2022-05-24 17:28:35 +00:00
Logger :: debug ( 'Delivered' , [ 'uri-id' => $uri_id , 'uid' => $uid , 'item_id' => $item_id , 'cmd' => $cmd , 'inbox' => $inbox , 'success' => $success , 'serverfailure' => $serverfail , 'drop' => $drop ]);
2022-05-19 19:24:21 +00:00
2022-05-24 17:28:35 +00:00
if (( $success || $drop ) && in_array ( $cmd , [ WorkerDelivery :: POST ])) {
2022-05-19 19:24:21 +00:00
Post\DeliveryData :: incrementQueueDone ( $uri_id , Post\DeliveryData :: ACTIVITYPUB );
}
2022-05-24 07:02:42 +00:00
return [ 'success' => $success , 'serverfailure' => $serverfail , 'drop' => $drop ];
}
/**
* Fetch the actor of the given inbox of an relay server
*
* @ param string $inbox
* @ return string
*/
private static function fetchActorForRelayInbox ( string $inbox ) : string
{
2022-05-24 08:06:48 +00:00
$apcontact = DBA :: selectFirst ( 'apcontact' , [ 'url' ], [ " `sharedinbox` = ? AND `type` = ? AND `url` IN (SELECT `url` FROM `contact` WHERE `uid` = ? AND `rel` = ?) " ,
$inbox , 'Application' , 0 , Contact :: FRIEND ]);
2022-05-24 08:02:55 +00:00
return $apcontact [ 'url' ] ? ? '' ;
2022-05-19 19:24:21 +00:00
}
2022-05-24 07:02:42 +00:00
/**
* mark or unmark the given receivers for archival upon succoess
*
* @ param array $receivers
* @ param boolean $success
* @ return void
*/
2022-05-19 19:24:21 +00:00
private static function setSuccess ( array $receivers , bool $success )
{
$gsid = null ;
foreach ( $receivers as $receiver ) {
$contact = Contact :: getById ( $receiver );
if ( empty ( $contact )) {
continue ;
}
$gsid = $gsid ? : $contact [ 'gsid' ];
if ( $success ) {
Contact :: unmarkForArchival ( $contact );
} else {
Contact :: markForArchival ( $contact );
}
}
if ( ! empty ( $gsid )) {
GServer :: setProtocol ( $gsid , Post\DeliveryData :: ACTIVITYPUB );
}
}
}