mirror of
https://github.com/pixelfed/pixelfed.git
synced 2025-01-11 22:50:45 +00:00
140 lines
4.1 KiB
PHP
140 lines
4.1 KiB
PHP
<?php
|
|
|
|
namespace App\Jobs\ProfilePipeline;
|
|
|
|
use App\Transformer\ActivityPub\Verb\Move;
|
|
use App\Util\ActivityPub\HttpSignature;
|
|
use GuzzleHttp\Client;
|
|
use GuzzleHttp\Pool;
|
|
use Illuminate\Bus\Batchable;
|
|
use Illuminate\Bus\Queueable;
|
|
use Illuminate\Contracts\Queue\ShouldBeUniqueUntilProcessing;
|
|
use Illuminate\Contracts\Queue\ShouldQueue;
|
|
use Illuminate\Foundation\Bus\Dispatchable;
|
|
use Illuminate\Queue\InteractsWithQueue;
|
|
use Illuminate\Queue\Middleware\WithoutOverlapping;
|
|
use Illuminate\Queue\SerializesModels;
|
|
use League\Fractal;
|
|
use League\Fractal\Serializer\ArraySerializer;
|
|
|
|
class ProfileMigrationDeliverMoveActivityPipeline implements ShouldBeUniqueUntilProcessing, ShouldQueue
|
|
{
|
|
use Batchable, Dispatchable, InteractsWithQueue, Queueable, SerializesModels;
|
|
|
|
public $migration;
|
|
|
|
public $oldAccount;
|
|
|
|
public $newAccount;
|
|
|
|
public $timeout = 1400;
|
|
|
|
public $tries = 3;
|
|
|
|
public $maxExceptions = 1;
|
|
|
|
public $failOnTimeout = true;
|
|
|
|
/**
|
|
* The number of seconds after which the job's unique lock will be released.
|
|
*
|
|
* @var int
|
|
*/
|
|
public $uniqueFor = 3600;
|
|
|
|
/**
|
|
* Get the unique ID for the job.
|
|
*/
|
|
public function uniqueId(): string
|
|
{
|
|
return 'profile:migration:deliver-move-followers:id:'.$this->migration->id;
|
|
}
|
|
|
|
/**
|
|
* Get the middleware the job should pass through.
|
|
*
|
|
* @return array<int, object>
|
|
*/
|
|
public function middleware(): array
|
|
{
|
|
return [(new WithoutOverlapping('profile:migration:deliver-move-followers:id:'.$this->migration->id))->shared()->dontRelease()];
|
|
}
|
|
|
|
/**
|
|
* Create a new job instance.
|
|
*/
|
|
public function __construct($migration, $oldAccount, $newAccount)
|
|
{
|
|
$this->migration = $migration;
|
|
$this->oldAccount = $oldAccount;
|
|
$this->newAccount = $newAccount;
|
|
}
|
|
|
|
/**
|
|
* Execute the job.
|
|
*/
|
|
public function handle(): void
|
|
{
|
|
if ($this->batch()->cancelled()) {
|
|
return;
|
|
}
|
|
|
|
$migration = $this->migration;
|
|
$profile = $this->oldAccount;
|
|
$newAccount = $this->newAccount;
|
|
|
|
if ($profile->domain || ! $profile->private_key) {
|
|
return;
|
|
}
|
|
|
|
$audience = $profile->getAudienceInbox();
|
|
$activitypubObject = new Move();
|
|
|
|
$fractal = new Fractal\Manager();
|
|
$fractal->setSerializer(new ArraySerializer());
|
|
$resource = new Fractal\Resource\Item($migration, $activitypubObject);
|
|
$activity = $fractal->createData($resource)->toArray();
|
|
|
|
$payload = json_encode($activity);
|
|
|
|
$client = new Client([
|
|
'timeout' => config('federation.activitypub.delivery.timeout'),
|
|
]);
|
|
|
|
$version = config('pixelfed.version');
|
|
$appUrl = config('app.url');
|
|
$userAgent = "(Pixelfed/{$version}; +{$appUrl})";
|
|
|
|
$requests = function ($audience) use ($client, $activity, $profile, $payload, $userAgent) {
|
|
foreach ($audience as $url) {
|
|
$headers = HttpSignature::sign($profile, $url, $activity, [
|
|
'Content-Type' => 'application/ld+json; profile="https://www.w3.org/ns/activitystreams"',
|
|
'User-Agent' => $userAgent,
|
|
]);
|
|
yield function () use ($client, $url, $headers, $payload) {
|
|
return $client->postAsync($url, [
|
|
'curl' => [
|
|
CURLOPT_HTTPHEADER => $headers,
|
|
CURLOPT_POSTFIELDS => $payload,
|
|
CURLOPT_HEADER => true,
|
|
CURLOPT_SSL_VERIFYPEER => false,
|
|
CURLOPT_SSL_VERIFYHOST => false,
|
|
],
|
|
]);
|
|
};
|
|
}
|
|
};
|
|
|
|
$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();
|
|
}
|
|
}
|