123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149 |
- <?php
- namespace App\Jobs\SharePipeline;
- use Cache, Log;
- use Illuminate\Support\Facades\Redis;
- use App\{Status, Notification};
- use Illuminate\Bus\Queueable;
- use Illuminate\Contracts\Queue\ShouldQueue;
- use Illuminate\Foundation\Bus\Dispatchable;
- use Illuminate\Queue\InteractsWithQueue;
- use Illuminate\Queue\SerializesModels;
- use League\Fractal;
- use League\Fractal\Serializer\ArraySerializer;
- use App\Transformer\ActivityPub\Verb\Announce;
- use GuzzleHttp\{Pool, Client, Promise};
- use App\Util\ActivityPub\HttpSignature;
- class SharePipeline implements ShouldQueue
- {
- 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;
- $actor = $status->profile;
- $target = $status->parent()->profile;
- if ($status->uri !== null) {
- // Ignore notifications to remote statuses
- return;
- }
- $exists = Notification::whereProfileId($target->id)
- ->whereActorId($status->profile_id)
- ->whereAction('share')
- ->whereItemId($status->reblog_of_id)
- ->whereItemType('App\Status')
- ->count();
- if ($target->id === $status->profile_id) {
- $this->remoteAnnounceDeliver();
- return true;
- }
- if( $exists !== 0) {
- return true;
- }
- $this->remoteAnnounceDeliver();
- try {
- $notification = new Notification;
- $notification->profile_id = $target->id;
- $notification->actor_id = $actor->id;
- $notification->action = 'share';
- $notification->message = $status->shareToText();
- $notification->rendered = $status->shareToHtml();
- $notification->item_id = $status->reblog_of_id ?? $status->id;
- $notification->item_type = "App\Status";
- $notification->save();
- $redis = Redis::connection();
- $key = config('cache.prefix').':user.'.$status->profile_id.'.notifications';
- $redis->lpush($key, $notification->id);
- } catch (Exception $e) {
- Log::error($e);
- }
- }
- public function remoteAnnounceDeliver()
- {
- if(config('federation.activitypub.enabled') == false) {
- 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')
- ]);
- $requests = function($audience) use ($client, $activity, $profile, $payload) {
- foreach($audience as $url) {
- $headers = HttpSignature::sign($profile, $url, $activity);
- 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();
- }
- }
|