pixelfed/app/Jobs/StatusPipeline/StatusDelete.php

208 lines
6.3 KiB
PHP
Raw Normal View History

2018-06-01 21:04:33 +00:00
<?php
namespace App\Jobs\StatusPipeline;
use DB, Storage;
2018-11-04 03:24:50 +00:00
use App\{
2020-12-14 09:58:07 +00:00
AccountInterstitial,
MediaTag,
2018-11-04 03:24:50 +00:00
Notification,
Report,
Status,
StatusHashtag,
};
2018-06-01 21:04:33 +00:00
use Illuminate\Bus\Queueable;
use Illuminate\Contracts\Queue\ShouldQueue;
use Illuminate\Foundation\Bus\Dispatchable;
2018-08-28 03:07:36 +00:00
use Illuminate\Queue\InteractsWithQueue;
use Illuminate\Queue\SerializesModels;
use League\Fractal;
use Illuminate\Support\Str;
use League\Fractal\Serializer\ArraySerializer;
use App\Transformer\ActivityPub\Verb\DeleteNote;
use App\Util\ActivityPub\Helpers;
use GuzzleHttp\Pool;
use GuzzleHttp\Client;
use GuzzleHttp\Promise;
use App\Util\ActivityPub\HttpSignature;
use App\Services\StatusService;
2018-06-01 21:04:33 +00:00
class StatusDelete implements ShouldQueue
{
use Dispatchable, InteractsWithQueue, Queueable, SerializesModels;
protected $status;
2019-01-12 22:19:57 +00:00
/**
* Delete the job if its models no longer exist.
*
* @var bool
*/
public $deleteWhenMissingModels = true;
2018-06-01 21:04:33 +00:00
/**
* 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;
2020-12-27 21:45:31 +00:00
$profile = $this->status->profile;
StatusService::del($status->id);
2020-12-27 21:45:31 +00:00
$count = $profile->statuses()
->getQuery()
->whereIn('type', ['photo', 'photo:album', 'video', 'video:album', 'photo:video:album'])
->whereNull('in_reply_to_id')
->whereNull('reblog_of_id')
->count();
$profile->status_count = ($count - 1);
$profile->save();
2018-11-22 20:21:36 +00:00
2019-06-09 23:27:04 +00:00
if(config('federation.activitypub.enabled') == true) {
$this->fanoutDelete($status);
} else {
$this->unlinkRemoveMedia($status);
}
2018-06-01 21:04:33 +00:00
}
public function unlinkRemoveMedia($status)
{
2018-08-28 03:07:36 +00:00
foreach ($status->media as $media) {
2018-06-01 21:04:33 +00:00
$thumbnail = storage_path("app/{$media->thumbnail_path}");
$photo = storage_path("app/{$media->media_path}");
try {
2018-08-28 03:07:36 +00:00
if (is_file($thumbnail)) {
2018-06-01 21:04:33 +00:00
unlink($thumbnail);
}
2018-08-28 03:07:36 +00:00
if (is_file($photo)) {
2018-06-01 21:04:33 +00:00
unlink($photo);
}
if( config('pixelfed.cloud_storage') == true) {
if( Str::of($media->media_path)
->startsWith('public/') &&
Storage::disk(config('filesystems.cloud'))
->exists($media->media_path)
) {
Storage::disk(config('filesystems.cloud'))
->delete($media->media_path);
}
if( Str::of($media->thumbnail_path)
->startsWith('public/') &&
Storage::disk(config('filesystems.cloud'))
->exists($media->thumbnail_path)
) {
Storage::disk(config('filesystems.cloud'))
->delete($media->thumbnail_path);
}
}
2018-06-01 21:04:33 +00:00
$media->delete();
} catch (Exception $e) {
}
}
2019-05-21 02:39:33 +00:00
if($status->in_reply_to_id) {
DB::transaction(function() use($status) {
$parent = Status::findOrFail($status->in_reply_to_id);
--$parent->reply_count;
$parent->save();
});
}
DB::transaction(function() use($status) {
$comments = Status::where('in_reply_to_id', $status->id)->get();
foreach ($comments as $comment) {
$comment->in_reply_to_id = null;
$comment->save();
Notification::whereItemType('App\Status')
->whereItemId($comment->id)
->delete();
}
$status->likes()->delete();
2018-06-14 03:29:20 +00:00
Notification::whereItemType('App\Status')
2019-05-21 02:39:33 +00:00
->whereItemId($status->id)
2018-06-14 03:29:20 +00:00
->delete();
2019-05-21 02:39:33 +00:00
StatusHashtag::whereStatusId($status->id)->delete();
Report::whereObjectType('App\Status')
->whereObjectId($status->id)
->delete();
MediaTag::where('status_id', $status->id)
->cursor()
->each(function($tag) {
Notification::where('item_type', 'App\MediaTag')
->where('item_id', $tag->id)
->forceDelete();
$tag->delete();
2020-12-14 09:58:07 +00:00
});
AccountInterstitial::where('item_type', 'App\Status')
->where('item_id', $status->id)
->delete();
2019-08-12 09:17:29 +00:00
$status->forceDelete();
2019-05-21 02:39:33 +00:00
});
2018-06-01 21:04:33 +00:00
return true;
}
protected function fanoutDelete($status)
{
$audience = $status->profile->getAudienceInbox();
$profile = $status->profile;
$fractal = new Fractal\Manager();
$fractal->setSerializer(new ArraySerializer());
$resource = new Fractal\Resource\Item($status, new DeleteNote());
$activity = $fractal->createData($resource)->toArray();
$this->unlinkRemoveMedia($status);
$payload = json_encode($activity);
$client = new Client([
2019-06-09 23:38:12 +00:00
'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), [
2019-06-09 23:38:12 +00:00
'concurrency' => config('federation.activitypub.delivery.concurrency'),
'fulfilled' => function ($response, $index) {
},
'rejected' => function ($reason, $index) {
}
]);
$promise = $pool->promise();
$promise->wait();
}
2018-06-01 21:04:33 +00:00
}