2018-11-04 03:24:06 +00:00
|
|
|
<?php
|
|
|
|
|
|
|
|
namespace App\Jobs\SharePipeline;
|
|
|
|
|
2019-12-11 06:04:03 +00:00
|
|
|
use Cache, Log;
|
|
|
|
use Illuminate\Support\Facades\Redis;
|
2019-06-24 05:50:42 +00:00
|
|
|
use App\{Status, Notification};
|
2018-11-04 03:24:06 +00:00
|
|
|
use Illuminate\Bus\Queueable;
|
|
|
|
use Illuminate\Contracts\Queue\ShouldQueue;
|
|
|
|
use Illuminate\Foundation\Bus\Dispatchable;
|
|
|
|
use Illuminate\Queue\InteractsWithQueue;
|
|
|
|
use Illuminate\Queue\SerializesModels;
|
2019-06-25 04:23:01 +00:00
|
|
|
use League\Fractal;
|
|
|
|
use League\Fractal\Serializer\ArraySerializer;
|
|
|
|
use App\Transformer\ActivityPub\Verb\Announce;
|
|
|
|
use GuzzleHttp\{Pool, Client, Promise};
|
|
|
|
use App\Util\ActivityPub\HttpSignature;
|
2022-03-11 06:34:34 +00:00
|
|
|
use App\Services\ReblogService;
|
2021-12-13 02:32:11 +00:00
|
|
|
use App\Services\StatusService;
|
2018-11-04 03:24:06 +00:00
|
|
|
|
|
|
|
class SharePipeline implements ShouldQueue
|
|
|
|
{
|
2021-05-12 00:17:03 +00:00
|
|
|
use Dispatchable, InteractsWithQueue, Queueable, SerializesModels;
|
|
|
|
|
|
|
|
protected $status;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Delete the job if its models no longer exist.
|
|
|
|
*
|
|
|
|
* @var bool
|
|
|
|
*/
|
|
|
|
public $deleteWhenMissingModels = true;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Create a new job instance.
|
|
|
|
*
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function __construct(Status $status)
|
|
|
|
{
|
|
|
|
$this->status = $status;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Execute the job.
|
|
|
|
*
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function handle()
|
|
|
|
{
|
|
|
|
$status = $this->status;
|
2023-07-30 11:45:14 +00:00
|
|
|
$parent = Status::find($this->status->reblog_of_id);
|
|
|
|
if(!$parent) {
|
|
|
|
return;
|
|
|
|
}
|
2021-05-12 00:17:03 +00:00
|
|
|
$actor = $status->profile;
|
2021-06-29 04:37:38 +00:00
|
|
|
$target = $parent->profile;
|
2021-05-12 00:17:03 +00:00
|
|
|
|
|
|
|
if ($status->uri !== null) {
|
|
|
|
// Ignore notifications to remote statuses
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-06-29 04:37:38 +00:00
|
|
|
if($target->id === $status->profile_id) {
|
2021-05-12 00:17:03 +00:00
|
|
|
$this->remoteAnnounceDeliver();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2023-01-31 10:18:25 +00:00
|
|
|
ReblogService::addPostReblog($parent->profile_id, $status->id);
|
2022-03-11 06:34:34 +00:00
|
|
|
|
2023-01-31 09:27:23 +00:00
|
|
|
$parent->reblogs_count = $parent->reblogs_count + 1;
|
2021-06-29 04:37:38 +00:00
|
|
|
$parent->save();
|
2021-12-15 04:13:30 +00:00
|
|
|
StatusService::del($parent->id);
|
2021-06-29 04:37:38 +00:00
|
|
|
|
2023-01-31 09:27:23 +00:00
|
|
|
Notification::firstOrCreate(
|
|
|
|
[
|
|
|
|
'profile_id' => $target->id,
|
|
|
|
'actor_id' => $actor->id,
|
|
|
|
'action' => 'share',
|
|
|
|
'item_type' => 'App\Status',
|
|
|
|
'item_id' => $status->reblog_of_id ?? $status->id,
|
|
|
|
]
|
|
|
|
);
|
|
|
|
|
|
|
|
return $this->remoteAnnounceDeliver();
|
2021-05-12 00:17:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public function remoteAnnounceDeliver()
|
|
|
|
{
|
2023-07-30 11:45:14 +00:00
|
|
|
if(config('app.env') !== 'production' || config_cache('federation.activitypub.enabled') == false) {
|
2021-05-12 00:17:03 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
$status = $this->status;
|
|
|
|
$profile = $status->profile;
|
|
|
|
|
|
|
|
$fractal = new Fractal\Manager();
|
|
|
|
$fractal->setSerializer(new ArraySerializer());
|
|
|
|
$resource = new Fractal\Resource\Item($status, new Announce());
|
|
|
|
$activity = $fractal->createData($resource)->toArray();
|
|
|
|
|
|
|
|
$audience = $status->profile->getAudienceInbox();
|
|
|
|
|
|
|
|
if(empty($audience) || $status->scope != 'public') {
|
|
|
|
// Return on profiles with no remote followers
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
$payload = json_encode($activity);
|
|
|
|
|
|
|
|
$client = new Client([
|
|
|
|
'timeout' => config('federation.activitypub.delivery.timeout')
|
|
|
|
]);
|
|
|
|
|
2023-01-31 07:15:41 +00:00
|
|
|
$version = config('pixelfed.version');
|
|
|
|
$appUrl = config('app.url');
|
|
|
|
$userAgent = "(Pixelfed/{$version}; +{$appUrl})";
|
|
|
|
|
|
|
|
$requests = function($audience) use ($client, $activity, $profile, $payload, $userAgent) {
|
2021-05-12 00:17:03 +00:00
|
|
|
foreach($audience as $url) {
|
2022-05-02 14:36:12 +00:00
|
|
|
$headers = HttpSignature::sign($profile, $url, $activity, [
|
|
|
|
'Content-Type' => 'application/ld+json; profile="https://www.w3.org/ns/activitystreams"',
|
2023-01-31 07:15:41 +00:00
|
|
|
'User-Agent' => $userAgent,
|
2022-05-02 14:36:12 +00:00
|
|
|
]);
|
2021-05-12 00:17:03 +00:00
|
|
|
yield function() use ($client, $url, $headers, $payload) {
|
|
|
|
return $client->postAsync($url, [
|
|
|
|
'curl' => [
|
|
|
|
CURLOPT_HTTPHEADER => $headers,
|
|
|
|
CURLOPT_POSTFIELDS => $payload,
|
|
|
|
CURLOPT_HEADER => true
|
|
|
|
]
|
|
|
|
]);
|
|
|
|
};
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
$pool = new Pool($client, $requests($audience), [
|
|
|
|
'concurrency' => config('federation.activitypub.delivery.concurrency'),
|
|
|
|
'fulfilled' => function ($response, $index) {
|
|
|
|
},
|
|
|
|
'rejected' => function ($reason, $index) {
|
|
|
|
}
|
|
|
|
]);
|
|
|
|
|
|
|
|
$promise = $pool->promise();
|
|
|
|
|
|
|
|
$promise->wait();
|
|
|
|
|
|
|
|
}
|
2018-11-04 03:24:06 +00:00
|
|
|
}
|