StatusDelete.php 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159
  1. <?php
  2. namespace App\Jobs\StatusPipeline;
  3. use DB;
  4. use App\{
  5. Notification,
  6. Report,
  7. Status,
  8. StatusHashtag,
  9. };
  10. use Illuminate\Bus\Queueable;
  11. use Illuminate\Contracts\Queue\ShouldQueue;
  12. use Illuminate\Foundation\Bus\Dispatchable;
  13. use Illuminate\Queue\InteractsWithQueue;
  14. use Illuminate\Queue\SerializesModels;
  15. use League\Fractal;
  16. use League\Fractal\Serializer\ArraySerializer;
  17. use App\Transformer\ActivityPub\Verb\DeleteNote;
  18. use App\Util\ActivityPub\Helpers;
  19. use GuzzleHttp\Pool;
  20. use GuzzleHttp\Client;
  21. use GuzzleHttp\Promise;
  22. use App\Util\ActivityPub\HttpSignature;
  23. class StatusDelete implements ShouldQueue
  24. {
  25. use Dispatchable, InteractsWithQueue, Queueable, SerializesModels;
  26. protected $status;
  27. /**
  28. * Delete the job if its models no longer exist.
  29. *
  30. * @var bool
  31. */
  32. public $deleteWhenMissingModels = true;
  33. /**
  34. * Create a new job instance.
  35. *
  36. * @return void
  37. */
  38. public function __construct(Status $status)
  39. {
  40. $this->status = $status;
  41. }
  42. /**
  43. * Execute the job.
  44. *
  45. * @return void
  46. */
  47. public function handle()
  48. {
  49. $status = $this->status;
  50. if(config('federation.activitypub.enabled') == true) {
  51. $this->fanoutDelete($status);
  52. } else {
  53. $this->unlinkRemoveMedia($status);
  54. }
  55. }
  56. public function unlinkRemoveMedia($status)
  57. {
  58. foreach ($status->media as $media) {
  59. $thumbnail = storage_path("app/{$media->thumbnail_path}");
  60. $photo = storage_path("app/{$media->media_path}");
  61. try {
  62. if (is_file($thumbnail)) {
  63. unlink($thumbnail);
  64. }
  65. if (is_file($photo)) {
  66. unlink($photo);
  67. }
  68. $media->delete();
  69. } catch (Exception $e) {
  70. }
  71. }
  72. if($status->in_reply_to_id) {
  73. DB::transaction(function() use($status) {
  74. $parent = Status::findOrFail($status->in_reply_to_id);
  75. --$parent->reply_count;
  76. $parent->save();
  77. });
  78. }
  79. DB::transaction(function() use($status) {
  80. $comments = Status::where('in_reply_to_id', $status->id)->get();
  81. foreach ($comments as $comment) {
  82. $comment->in_reply_to_id = null;
  83. $comment->save();
  84. Notification::whereItemType('App\Status')
  85. ->whereItemId($comment->id)
  86. ->delete();
  87. }
  88. $status->likes()->delete();
  89. Notification::whereItemType('App\Status')
  90. ->whereItemId($status->id)
  91. ->delete();
  92. StatusHashtag::whereStatusId($status->id)->delete();
  93. Report::whereObjectType('App\Status')
  94. ->whereObjectId($status->id)
  95. ->delete();
  96. $status->forceDelete();
  97. });
  98. return true;
  99. }
  100. protected function fanoutDelete($status)
  101. {
  102. $audience = $status->profile->getAudienceInbox();
  103. $profile = $status->profile;
  104. $fractal = new Fractal\Manager();
  105. $fractal->setSerializer(new ArraySerializer());
  106. $resource = new Fractal\Resource\Item($status, new DeleteNote());
  107. $activity = $fractal->createData($resource)->toArray();
  108. $this->unlinkRemoveMedia($status);
  109. $payload = json_encode($activity);
  110. $client = new Client([
  111. 'timeout' => config('federation.activitypub.delivery.timeout')
  112. ]);
  113. $requests = function($audience) use ($client, $activity, $profile, $payload) {
  114. foreach($audience as $url) {
  115. $headers = HttpSignature::sign($profile, $url, $activity);
  116. yield function() use ($client, $url, $headers, $payload) {
  117. return $client->postAsync($url, [
  118. 'curl' => [
  119. CURLOPT_HTTPHEADER => $headers,
  120. CURLOPT_POSTFIELDS => $payload,
  121. CURLOPT_HEADER => true
  122. ]
  123. ]);
  124. };
  125. }
  126. };
  127. $pool = new Pool($client, $requests($audience), [
  128. 'concurrency' => config('federation.activitypub.delivery.concurrency'),
  129. 'fulfilled' => function ($response, $index) {
  130. },
  131. 'rejected' => function ($reason, $index) {
  132. }
  133. ]);
  134. $promise = $pool->promise();
  135. $promise->wait();
  136. }
  137. }