123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440 |
- <?php
- namespace App\Util\ActivityPub;
- use App\DirectMessage;
- use App\Follower;
- use App\FollowRequest;
- use App\Instance;
- use App\Jobs\DeletePipeline\DeleteRemoteProfilePipeline;
- use App\Jobs\FollowPipeline\FollowPipeline;
- use App\Jobs\HomeFeedPipeline\FeedRemoveRemotePipeline;
- use App\Jobs\LikePipeline\LikePipeline;
- use App\Jobs\MovePipeline\CleanupLegacyAccountMovePipeline;
- use App\Jobs\MovePipeline\MoveMigrateFollowersPipeline;
- use App\Jobs\MovePipeline\ProcessMovePipeline;
- use App\Jobs\MovePipeline\UnfollowLegacyAccountMovePipeline;
- use App\Jobs\ProfilePipeline\HandleUpdateActivity;
- use App\Jobs\PushNotificationPipeline\MentionPushNotifyPipeline;
- use App\Jobs\StatusPipeline\RemoteStatusDelete;
- use App\Jobs\StatusPipeline\StatusRemoteUpdatePipeline;
- use App\Jobs\StoryPipeline\StoryExpire;
- use App\Jobs\StoryPipeline\StoryFetch;
- use App\Like;
- use App\Media;
- use App\Models\Conversation;
- use App\Models\RemoteReport;
- use App\Notification;
- use App\Profile;
- use App\Services\AccountService;
- use App\Services\FollowerService;
- use App\Services\NotificationAppGatewayService;
- use App\Services\PollService;
- use App\Services\PushNotificationService;
- use App\Services\ReblogService;
- use App\Services\RelationshipService;
- use App\Services\SanitizeService;
- use App\Services\StoryIndexService;
- use App\Services\UserFilterService;
- use App\Status;
- use App\Story;
- use App\StoryView;
- use App\UserFilter;
- use App\Util\ActivityPub\Validator\Accept as AcceptValidator;
- use App\Util\ActivityPub\Validator\Announce as AnnounceValidator;
- use App\Util\ActivityPub\Validator\Follow as FollowValidator;
- use App\Util\ActivityPub\Validator\Like as LikeValidator;
- use App\Util\ActivityPub\Validator\MoveValidator;
- use App\Util\ActivityPub\Validator\RejectValidator;
- use App\Util\ActivityPub\Validator\UpdatePersonValidator;
- use Cache;
- use Illuminate\Support\Facades\Bus;
- use Illuminate\Support\Facades\Log;
- use Illuminate\Support\Str;
- use Storage;
- use Throwable;
- class Inbox
- {
- protected $headers;
- protected $profile;
- protected $payload;
- protected $logger;
- public function __construct($headers, $profile, $payload)
- {
- $this->headers = $headers;
- $this->profile = $profile;
- $this->payload = $payload;
- }
- public function handle()
- {
- $this->handleVerb();
- }
- public function handleVerb()
- {
- $verb = (string) $this->payload['type'];
- switch ($verb) {
- case 'Add':
- $this->handleAddActivity();
- break;
- case 'Create':
- $this->handleCreateActivity();
- break;
- case 'Follow':
- if (FollowValidator::validate($this->payload) == false) {
- return;
- }
- $this->handleFollowActivity();
- break;
- case 'Announce':
- if (AnnounceValidator::validate($this->payload) == false) {
- return;
- }
- $this->handleAnnounceActivity();
- break;
- case 'Accept':
- if (AcceptValidator::validate($this->payload) == false) {
- return;
- }
- $this->handleAcceptActivity();
- break;
- case 'Delete':
- $this->handleDeleteActivity();
- break;
- case 'Like':
- if (LikeValidator::validate($this->payload) == false) {
- return;
- }
- $this->handleLikeActivity();
- break;
- case 'Reject':
- if (RejectValidator::validate($this->payload) == false) {
- return;
- }
- $this->handleRejectActivity();
- break;
- case 'Undo':
- $this->handleUndoActivity();
- break;
- case 'View':
- $this->handleViewActivity();
- break;
- case 'Story:Reaction':
- $this->handleStoryReactionActivity();
- break;
- case 'Story:Reply':
- $this->handleStoryReplyActivity();
- break;
- case 'Flag':
- $this->handleFlagActivity();
- break;
- case 'Update':
- $this->handleUpdateActivity();
- break;
- case 'Move':
- if (MoveValidator::validate($this->payload) == false) {
- Log::info('[AP][INBOX][MOVE] VALIDATE_FAILURE '.json_encode($this->payload));
- return;
- }
- $this->handleMoveActivity();
- break;
- default:
- // TODO: decide how to handle invalid verbs.
- break;
- }
- }
- public function verifyNoteAttachment()
- {
- $activity = $this->payload['object'];
- if (isset($activity['inReplyTo']) &&
- ! empty($activity['inReplyTo']) &&
- Helpers::validateUrl($activity['inReplyTo'])
- ) {
- // reply detected, skip attachment check
- return true;
- }
- $valid = Helpers::verifyAttachments($activity);
- return $valid;
- }
- public function actorFirstOrCreate($actorUrl)
- {
- return Helpers::profileFetch($actorUrl);
- }
- public function handleAddActivity()
- {
- // stories ;)
- if (! isset(
- $this->payload['actor'],
- $this->payload['object']
- )) {
- return;
- }
- $actor = $this->payload['actor'];
- $obj = $this->payload['object'];
- if (! Helpers::validateUrl($actor)) {
- return;
- }
- if (! isset($obj['type'])) {
- return;
- }
- switch ($obj['type']) {
- case 'Story':
- StoryFetch::dispatch($this->payload);
- break;
- }
- }
- public function handleCreateActivity()
- {
- $activity = $this->payload['object'];
- if (config('autospam.live_filters.enabled')) {
- $filters = config('autospam.live_filters.filters');
- if (! empty($filters) && isset($activity['content']) && ! empty($activity['content']) && strlen($filters) > 3) {
- $filters = array_map('trim', explode(',', $filters));
- $content = $activity['content'];
- foreach ($filters as $filter) {
- $filter = trim(strtolower($filter));
- if (! $filter || ! strlen($filter)) {
- continue;
- }
- if (str_contains(strtolower($content), $filter)) {
- return;
- }
- }
- }
- }
- $actor = $this->actorFirstOrCreate($this->payload['actor']);
- if (! $actor || $actor->domain == null) {
- return;
- }
- if (! isset($activity['to'])) {
- return;
- }
- $to = isset($activity['to']) ? $activity['to'] : [];
- $cc = isset($activity['cc']) ? $activity['cc'] : [];
- if ($activity['type'] == 'Question') {
- // $this->handlePollCreate();
- return;
- }
- if (is_array($to) &&
- is_array($cc) &&
- count($to) == 1 &&
- count($cc) == 0 &&
- parse_url($to[0], PHP_URL_HOST) == config('pixelfed.domain.app')
- ) {
- $this->handleDirectMessage();
- return;
- }
- if ($activity['type'] == 'Note' && ! empty($activity['inReplyTo'])) {
- $this->handleNoteReply();
- } elseif ($activity['type'] == 'Note' && ! empty($activity['attachment'])) {
- if (! $this->verifyNoteAttachment()) {
- return;
- }
- $this->handleNoteCreate();
- }
- }
- public function handleNoteReply()
- {
- $activity = $this->payload['object'];
- $actor = $this->actorFirstOrCreate($this->payload['actor']);
- if (! $actor || $actor->domain == null) {
- return;
- }
- $inReplyTo = $activity['inReplyTo'];
- $url = isset($activity['url']) ? $activity['url'] : $activity['id'];
- Helpers::statusFirstOrFetch($url, true);
- }
- public function handlePollCreate()
- {
- $activity = $this->payload['object'];
- $actor = $this->actorFirstOrCreate($this->payload['actor']);
- if (! $actor || $actor->domain == null) {
- return;
- }
- $url = isset($activity['url']) ? $activity['url'] : $activity['id'];
- Helpers::statusFirstOrFetch($url);
- }
- public function handleNoteCreate()
- {
- $activity = $this->payload['object'];
- $actor = $this->actorFirstOrCreate($this->payload['actor']);
- if (! $actor || $actor->domain == null) {
- return;
- }
- if (isset($activity['inReplyTo']) &&
- isset($activity['name']) &&
- ! isset($activity['content']) &&
- ! isset($activity['attachment']) &&
- Helpers::validateLocalUrl($activity['inReplyTo'])
- ) {
- $this->handlePollVote();
- return;
- }
- if ($actor->followers_count == 0) {
- if (config('federation.activitypub.ingest.store_notes_without_followers')) {
- } elseif (FollowerService::followerCount($actor->id, true) == 0) {
- return;
- }
- }
- $hasUrl = isset($activity['url']);
- $url = isset($activity['url']) ? $activity['url'] : $activity['id'];
- if ($hasUrl) {
- if (Status::whereUri($url)->exists()) {
- return;
- }
- } else {
- if (Status::whereObjectUrl($url)->exists()) {
- return;
- }
- }
- Helpers::storeStatus(
- $url,
- $actor,
- $activity
- );
- }
- public function handlePollVote()
- {
- $activity = $this->payload['object'];
- $actor = $this->actorFirstOrCreate($this->payload['actor']);
- if (! $actor) {
- return;
- }
- $status = Helpers::statusFetch($activity['inReplyTo']);
- if (! $status) {
- return;
- }
- $poll = $status->poll;
- if (! $poll) {
- return;
- }
- if (now()->gt($poll->expires_at)) {
- return;
- }
- $choices = $poll->poll_options;
- $choice = array_search($activity['name'], $choices);
- if ($choice === false) {
- return;
- }
- if (PollVote::whereStatusId($status->id)->whereProfileId($actor->id)->exists()) {
- return;
- }
- $vote = new PollVote;
- $vote->status_id = $status->id;
- $vote->profile_id = $actor->id;
- $vote->poll_id = $poll->id;
- $vote->choice = $choice;
- $vote->uri = isset($activity['id']) ? $activity['id'] : null;
- $vote->save();
- $tallies = $poll->cached_tallies;
- $tallies[$choice] = $tallies[$choice] + 1;
- $poll->cached_tallies = $tallies;
- $poll->votes_count = array_sum($tallies);
- $poll->save();
- PollService::del($status->id);
- }
- public function handleDirectMessage()
- {
- $activity = $this->payload['object'];
- $actor = $this->actorFirstOrCreate($this->payload['actor']);
- $profile = Profile::whereNull('domain')
- ->whereUsername(array_last(explode('/', $activity['to'][0])))
- ->firstOrFail();
- if (! $actor || in_array($actor->id, $profile->blockedIds()->toArray())) {
- return;
- }
- if (AccountService::blocksDomain($profile->id, $actor->domain) == true) {
- return;
- }
- $msg = app(SanitizeService::class)->html($activity['content']);
- $msgText = strip_tags($msg);
- if (Str::startsWith($msgText, '@'.$profile->username)) {
- $len = strlen('@'.$profile->username);
- $msgText = substr($msgText, $len + 1);
- }
- if ($profile->user->settings->public_dm == false || $profile->is_private) {
- if ($profile->follows($actor) == true) {
- $hidden = false;
- } else {
- $hidden = true;
- }
- } else {
- $hidden = false;
- }
- $status = new Status;
- $status->profile_id = $actor->id;
- $status->caption = $msgText;
- $status->visibility = 'direct';
- $status->scope = 'direct';
- $status->url = $activity['id'];
- $status->uri = $activity['id'];
- $status->object_url = $activity['id'];
- $status->in_reply_to_profile_id = $profile->id;
- $status->save();
- $dm = new DirectMessage;
- $dm->to_id = $profile->id;
- $dm->from_id = $actor->id;
- $dm->status_id = $status->id;
- $dm->is_hidden = $hidden;
- $dm->type = 'text';
- $dm->save();
- Conversation::updateOrInsert(
- [
- 'to_id' => $profile->id,
- 'from_id' => $actor->id,
- ],
- [
- 'type' => 'text',
- 'status_id' => $status->id,
- 'dm_id' => $dm->id,
- 'is_hidden' => $hidden,
- ]
- );
- if (count($activity['attachment'])) {
- $photos = 0;
- $videos = 0;
- $allowed = explode(',', config_cache('pixelfed.media_types'));
- $activity['attachment'] = array_slice($activity['attachment'], 0, config_cache('pixelfed.max_album_length'));
- foreach ($activity['attachment'] as $a) {
- $type = $a['mediaType'];
- $url = $a['url'];
- $valid = Helpers::validateUrl($url);
- if (in_array($type, $allowed) == false || $valid == false) {
- continue;
- }
- $media = new Media;
- $media->remote_media = true;
- $media->status_id = $status->id;
- $media->profile_id = $status->profile_id;
- $media->user_id = null;
- $media->media_path = $url;
- $media->remote_url = $url;
- $media->mime = $type;
- $media->save();
- if (explode('/', $type)[0] == 'image') {
- $photos = $photos + 1;
- }
- if (explode('/', $type)[0] == 'video') {
- $videos = $videos + 1;
- }
- }
- if ($photos && $videos == 0) {
- $dm->type = $photos == 1 ? 'photo' : 'photos';
- $dm->save();
- }
- if ($videos && $photos == 0) {
- $dm->type = $videos == 1 ? 'video' : 'videos';
- $dm->save();
- }
- }
- if (filter_var($msgText, FILTER_VALIDATE_URL)) {
- if (Helpers::validateUrl($msgText)) {
- $dm->type = 'link';
- $dm->meta = [
- 'domain' => parse_url($msgText, PHP_URL_HOST),
- 'local' => parse_url($msgText, PHP_URL_HOST) ==
- parse_url(config('app.url'), PHP_URL_HOST),
- ];
- $dm->save();
- }
- }
- $nf = UserFilter::whereUserId($profile->id)
- ->whereFilterableId($actor->id)
- ->whereFilterableType('App\Profile')
- ->whereFilterType('dm.mute')
- ->exists();
- if ($profile->domain == null && $hidden == false && ! $nf) {
- $notification = new Notification;
- $notification->profile_id = $profile->id;
- $notification->actor_id = $actor->id;
- $notification->action = 'dm';
- $notification->item_id = $dm->id;
- $notification->item_type = "App\DirectMessage";
- $notification->save();
- if (NotificationAppGatewayService::enabled()) {
- if (PushNotificationService::check('mention', $profile->id)) {
- $user = User::whereProfileId($profile->id)->first();
- if ($user && $user->expo_token && $user->notify_enabled) {
- MentionPushNotifyPipeline::dispatch($user->expo_token, $actor->username)->onQueue('pushnotify');
- }
- }
- }
- }
- }
- public function handleFollowActivity()
- {
- $actor = $this->actorFirstOrCreate($this->payload['actor']);
- $target = $this->actorFirstOrCreate($this->payload['object']);
- if (! $actor || ! $target) {
- return;
- }
- if ($actor->domain == null || $target->domain !== null) {
- return;
- }
- if (AccountService::blocksDomain($target->id, $actor->domain) == true) {
- return;
- }
- if (
- Follower::whereProfileId($actor->id)
- ->whereFollowingId($target->id)
- ->exists() ||
- FollowRequest::whereFollowerId($actor->id)
- ->whereFollowingId($target->id)
- ->exists()
- ) {
- return;
- }
- $blocks = UserFilterService::blocks($target->id);
- if ($blocks && in_array($actor->id, $blocks)) {
- return;
- }
- if ($target->is_private == true) {
- FollowRequest::updateOrCreate([
- 'follower_id' => $actor->id,
- 'following_id' => $target->id,
- ], [
- 'activity' => collect($this->payload)->only(['id', 'actor', 'object', 'type'])->toArray(),
- ]);
- } else {
- $follower = new Follower;
- $follower->profile_id = $actor->id;
- $follower->following_id = $target->id;
- $follower->local_profile = empty($actor->domain);
- $follower->save();
- FollowPipeline::dispatch($follower);
- FollowerService::add($actor->id, $target->id);
- // send Accept to remote profile
- $accept = [
- '@context' => 'https://www.w3.org/ns/activitystreams',
- 'id' => $target->permalink().'#accepts/follows/'.$follower->id,
- 'type' => 'Accept',
- 'actor' => $target->permalink(),
- 'object' => [
- 'id' => $this->payload['id'],
- 'actor' => $actor->permalink(),
- 'type' => 'Follow',
- 'object' => $target->permalink(),
- ],
- ];
- Helpers::sendSignedObject($target, $actor->inbox_url, $accept);
- Cache::forget('profile:follower_count:'.$target->id);
- Cache::forget('profile:follower_count:'.$actor->id);
- Cache::forget('profile:following_count:'.$target->id);
- Cache::forget('profile:following_count:'.$actor->id);
- }
- RelationshipService::refresh($actor->id, $target->id);
- AccountService::del($actor->id);
- AccountService::del($target->id);
- }
- public function handleAnnounceActivity()
- {
- $actor = $this->actorFirstOrCreate($this->payload['actor']);
- $activity = $this->payload['object'];
- if (! $actor || $actor->domain == null) {
- return;
- }
- $parent = Helpers::statusFetch($activity);
- if (! $parent || empty($parent)) {
- return;
- }
- if (AccountService::blocksDomain($parent->profile_id, $actor->domain) == true) {
- return;
- }
- $blocks = UserFilterService::blocks($parent->profile_id);
- if ($blocks && in_array($actor->id, $blocks)) {
- return;
- }
- $status = Status::firstOrCreate([
- 'profile_id' => $actor->id,
- 'reblog_of_id' => $parent->id,
- 'type' => 'share',
- 'local' => false,
- ]);
- Notification::firstOrCreate(
- [
- 'profile_id' => $parent->profile_id,
- 'actor_id' => $actor->id,
- 'action' => 'share',
- 'item_id' => $parent->id,
- 'item_type' => 'App\Status',
- ]
- );
- $parent->reblogs_count = $parent->reblogs_count + 1;
- $parent->save();
- ReblogService::addPostReblog($parent->profile_id, $status->id);
- }
- public function handleAcceptActivity()
- {
- $actor = $this->payload['object']['actor'];
- $obj = $this->payload['object']['object'];
- $type = $this->payload['object']['type'];
- if ($type !== 'Follow') {
- return;
- }
- $actor = Helpers::validateLocalUrl($actor);
- $target = Helpers::validateUrl($obj);
- if (! $actor || ! $target) {
- return;
- }
- $actor = Helpers::profileFetch($actor);
- $target = Helpers::profileFetch($target);
- if (! $actor || ! $target) {
- return;
- }
- if (AccountService::blocksDomain($target->id, $actor->domain) == true) {
- return;
- }
- $request = FollowRequest::whereFollowerId($actor->id)
- ->whereFollowingId($target->id)
- ->whereIsRejected(false)
- ->first();
- if (! $request) {
- return;
- }
- $follower = Follower::firstOrCreate([
- 'profile_id' => $actor->id,
- 'following_id' => $target->id,
- ]);
- FollowPipeline::dispatch($follower)->onQueue('high');
- RelationshipService::refresh($actor->id, $target->id);
- Cache::forget('profile:following:'.$target->id);
- Cache::forget('profile:followers:'.$target->id);
- Cache::forget('profile:following:'.$actor->id);
- Cache::forget('profile:followers:'.$actor->id);
- Cache::forget('profile:follower_count:'.$target->id);
- Cache::forget('profile:follower_count:'.$actor->id);
- Cache::forget('profile:following_count:'.$target->id);
- Cache::forget('profile:following_count:'.$actor->id);
- AccountService::del($actor->id);
- AccountService::del($target->id);
- RelationshipService::get($actor->id, $target->id);
- $request->delete();
- }
- public function handleDeleteActivity()
- {
- if (! isset(
- $this->payload['actor'],
- $this->payload['object']
- )) {
- return;
- }
- $actor = $this->payload['actor'];
- $obj = $this->payload['object'];
- if (is_string($obj) == true && $actor == $obj && Helpers::validateUrl($obj)) {
- $profile = Profile::whereRemoteUrl($obj)->first();
- if (! $profile || $profile->private_key != null) {
- return;
- }
- DeleteRemoteProfilePipeline::dispatch($profile)->onQueue('inbox');
- return;
- } else {
- if (! isset(
- $obj['id'],
- $this->payload['object'],
- $this->payload['object']['id'],
- $this->payload['object']['type']
- )) {
- return;
- }
- $type = $this->payload['object']['type'];
- $typeCheck = in_array($type, ['Person', 'Tombstone', 'Story']);
- if (! Helpers::validateUrl($actor) || ! Helpers::validateUrl($obj['id']) || ! $typeCheck) {
- return;
- }
- if (parse_url($obj['id'], PHP_URL_HOST) !== parse_url($actor, PHP_URL_HOST)) {
- return;
- }
- $id = $this->payload['object']['id'];
- switch ($type) {
- case 'Person':
- $profile = Profile::whereRemoteUrl($actor)->first();
- if (! $profile || $profile->private_key != null) {
- return;
- }
- DeleteRemoteProfilePipeline::dispatch($profile)->onQueue('inbox');
- return;
- break;
- case 'Tombstone':
- $profile = Profile::whereRemoteUrl($actor)->first();
- if (! $profile || $profile->private_key != null) {
- return;
- }
- $status = Status::where('object_url', $id)->first();
- if (! $status) {
- $status = Status::where('url', $id)->first();
- if (! $status) {
- return;
- }
- }
- if ($status->profile_id != $profile->id) {
- return;
- }
- if ($status->scope && in_array($status->scope, ['public', 'unlisted', 'private'])) {
- if ($status->type && ! in_array($status->type, ['story:reaction', 'story:reply', 'reply'])) {
- FeedRemoveRemotePipeline::dispatch($status->id, $status->profile_id)->onQueue('feed');
- }
- }
- RemoteStatusDelete::dispatch($status)->onQueue('high');
- return;
- break;
- case 'Story':
- $story = Story::whereObjectId($id)
- ->first();
- if ($story) {
- StoryExpire::dispatch($story)->onQueue('story');
- }
- return;
- break;
- default:
- return;
- break;
- }
- }
- }
- public function handleLikeActivity()
- {
- $actor = $this->payload['actor'];
- if (! Helpers::validateUrl($actor)) {
- return;
- }
- $profile = self::actorFirstOrCreate($actor);
- $obj = $this->payload['object'];
- if (! Helpers::validateUrl($obj)) {
- return;
- }
- $status = Helpers::statusFirstOrFetch($obj);
- if (! $status || ! $profile) {
- return;
- }
- if (AccountService::blocksDomain($status->profile_id, $profile->domain) == true) {
- return;
- }
- $blocks = UserFilterService::blocks($status->profile_id);
- if ($blocks && in_array($profile->id, $blocks)) {
- return;
- }
- $like = Like::firstOrCreate([
- 'profile_id' => $profile->id,
- 'status_id' => $status->id,
- ]);
- if ($like->wasRecentlyCreated == true) {
- $status->likes_count = $status->likes_count + 1;
- $status->save();
- LikePipeline::dispatch($like);
- }
- }
- public function handleRejectActivity()
- {
- $actorUrl = $this->payload['actor'];
- $obj = $this->payload['object'];
- $profileUrl = $obj['actor'];
- if (! Helpers::validateUrl($actorUrl) || ! Helpers::validateLocalUrl($profileUrl)) {
- return;
- }
- $actor = Helpers::profileFetch($actorUrl);
- $profile = Helpers::profileFetch($profileUrl);
- FollowRequest::whereFollowerId($profile->id)->whereFollowingId($actor->id)->forceDelete();
- RelationshipService::refresh($actor->id, $profile->id);
- }
- public function handleUndoActivity()
- {
- $actor = $this->payload['actor'];
- $profile = self::actorFirstOrCreate($actor);
- $obj = $this->payload['object'];
- if (! $profile) {
- return;
- }
- // TODO: Some implementations do not inline the object, skip for now
- if (! $obj || ! is_array($obj) || ! isset($obj['type'])) {
- return;
- }
- switch ($obj['type']) {
- case 'Accept':
- break;
- case 'Announce':
- if (is_array($obj) && isset($obj['object'])) {
- $obj = $obj['object'];
- }
- if (! is_string($obj)) {
- return;
- }
- if (Helpers::validateLocalUrl($obj)) {
- $parsedId = last(explode('/', $obj));
- $status = Status::find($parsedId);
- } else {
- $status = Status::whereUri($obj)->first();
- }
- if (! $status) {
- return;
- }
- if (AccountService::blocksDomain($status->profile_id, $profile->domain) == true) {
- return;
- }
- FeedRemoveRemotePipeline::dispatch($status->id, $status->profile_id)->onQueue('feed');
- Status::whereProfileId($profile->id)
- ->whereReblogOfId($status->id)
- ->delete();
- ReblogService::removePostReblog($profile->id, $status->id);
- Notification::whereProfileId($status->profile_id)
- ->whereActorId($profile->id)
- ->whereAction('share')
- ->whereItemId($status->reblog_of_id)
- ->whereItemType('App\Status')
- ->forceDelete();
- break;
- case 'Block':
- break;
- case 'Follow':
- $following = self::actorFirstOrCreate($obj['object']);
- if (! $following) {
- return;
- }
- if (AccountService::blocksDomain($following->id, $profile->domain) == true) {
- return;
- }
- Follower::whereProfileId($profile->id)
- ->whereFollowingId($following->id)
- ->delete();
- FollowRequest::whereFollowingId($following->id)
- ->whereFollowerId($profile->id)
- ->forceDelete();
- Notification::whereProfileId($following->id)
- ->whereActorId($profile->id)
- ->whereAction('follow')
- ->whereItemId($following->id)
- ->whereItemType('App\Profile')
- ->forceDelete();
- FollowerService::remove($profile->id, $following->id);
- RelationshipService::refresh($following->id, $profile->id);
- AccountService::del($profile->id);
- AccountService::del($following->id);
- break;
- case 'Like':
- $objectUri = $obj['object'];
- if (! is_string($objectUri)) {
- if (is_array($objectUri) && isset($objectUri['id']) && is_string($objectUri['id'])) {
- $objectUri = $objectUri['id'];
- } else {
- return;
- }
- }
- $status = Helpers::statusFirstOrFetch($objectUri);
- if (! $status) {
- return;
- }
- if (AccountService::blocksDomain($status->profile_id, $profile->domain) == true) {
- return;
- }
- Like::whereProfileId($profile->id)
- ->whereStatusId($status->id)
- ->forceDelete();
- Notification::whereProfileId($status->profile_id)
- ->whereActorId($profile->id)
- ->whereAction('like')
- ->whereItemId($status->id)
- ->whereItemType('App\Status')
- ->forceDelete();
- break;
- }
- }
- public function handleViewActivity()
- {
- if (! isset(
- $this->payload['actor'],
- $this->payload['object']
- )) {
- return;
- }
- $actor = $this->payload['actor'];
- $obj = $this->payload['object'];
- if (! Helpers::validateUrl($actor)) {
- return;
- }
- if (! $obj || ! is_array($obj)) {
- return;
- }
- if (! isset($obj['type']) || ! isset($obj['object']) || $obj['type'] != 'Story') {
- return;
- }
- if (! Helpers::validateLocalUrl($obj['object'])) {
- return;
- }
- $profile = Helpers::profileFetch($actor);
- $storyId = Str::of($obj['object'])->explode('/')->last();
- $story = Story::whereActive(true)
- ->whereLocal(true)
- ->find($storyId);
- if (! $story) {
- return;
- }
- if (AccountService::blocksDomain($story->profile_id, $profile->domain) == true) {
- return;
- }
- if (! FollowerService::follows($profile->id, $story->profile_id)) {
- return;
- }
- $view = StoryView::firstOrCreate([
- 'story_id' => $story->id,
- 'profile_id' => $profile->id,
- ]);
- $index = app(StoryIndexService::class);
- $index->markSeen($profile->id, $story->profile_id, $story->id, $story->created_at);
- if ($view->wasRecentlyCreated == true) {
- $story->view_count++;
- $story->save();
- }
- }
- public function handleStoryReactionActivity()
- {
- if (! isset(
- $this->payload['actor'],
- $this->payload['id'],
- $this->payload['inReplyTo'],
- $this->payload['content']
- )) {
- return;
- }
- $id = $this->payload['id'];
- $actor = $this->payload['actor'];
- $storyUrl = $this->payload['inReplyTo'];
- $to = $this->payload['to'];
- $text = app(SanitizeService::class)->html($this->payload['content']);
- if (parse_url($id, PHP_URL_HOST) !== parse_url($actor, PHP_URL_HOST)) {
- return;
- }
- if (! Helpers::validateUrl($id) || ! Helpers::validateUrl($actor)) {
- return;
- }
- if (! Helpers::validateLocalUrl($storyUrl)) {
- return;
- }
- if (! Helpers::validateLocalUrl($to)) {
- return;
- }
- if (Status::whereObjectUrl($id)->exists()) {
- return;
- }
- $storyId = Str::of($storyUrl)->explode('/')->last();
- $targetProfile = Helpers::profileFetch($to);
- $story = Story::whereProfileId($targetProfile->id)
- ->find($storyId);
- if (! $story) {
- return;
- }
- if ($story->can_react == false) {
- return;
- }
- $actorProfile = Helpers::profileFetch($actor);
- if (AccountService::blocksDomain($targetProfile->id, $actorProfile->domain) == true) {
- return;
- }
- if (! FollowerService::follows($actorProfile->id, $targetProfile->id)) {
- return;
- }
- $url = $id;
- if (str_ends_with($url, '/activity')) {
- $url = substr($url, 0, -9);
- }
- $status = new Status;
- $status->profile_id = $actorProfile->id;
- $status->type = 'story:reaction';
- $status->url = $url;
- $status->uri = $url;
- $status->object_url = $url;
- $status->caption = $text;
- $status->scope = 'direct';
- $status->visibility = 'direct';
- $status->in_reply_to_profile_id = $story->profile_id;
- $status->entities = json_encode([
- 'story_id' => $story->id,
- 'reaction' => $text,
- ]);
- $status->save();
- $dm = new DirectMessage;
- $dm->to_id = $story->profile_id;
- $dm->from_id = $actorProfile->id;
- $dm->type = 'story:react';
- $dm->status_id = $status->id;
- $dm->meta = json_encode([
- 'story_username' => $targetProfile->username,
- 'story_actor_username' => $actorProfile->username,
- 'story_id' => $story->id,
- 'story_media_url' => url(Storage::url($story->path)),
- 'reaction' => $text,
- ]);
- $dm->save();
- Conversation::updateOrInsert(
- [
- 'to_id' => $story->profile_id,
- 'from_id' => $actorProfile->id,
- ],
- [
- 'type' => 'story:react',
- 'status_id' => $status->id,
- 'dm_id' => $dm->id,
- 'is_hidden' => false,
- ]
- );
- $n = new Notification;
- $n->profile_id = $dm->to_id;
- $n->actor_id = $dm->from_id;
- $n->item_id = $dm->id;
- $n->item_type = 'App\DirectMessage';
- $n->action = 'story:react';
- $n->save();
- }
- public function handleStoryReplyActivity()
- {
- if (! isset(
- $this->payload['actor'],
- $this->payload['id'],
- $this->payload['inReplyTo'],
- $this->payload['content']
- )) {
- return;
- }
- $id = $this->payload['id'];
- $actor = $this->payload['actor'];
- $storyUrl = $this->payload['inReplyTo'];
- $to = $this->payload['to'];
- $text = app(SanitizeService::class)->html($this->payload['content']);
- if (parse_url($id, PHP_URL_HOST) !== parse_url($actor, PHP_URL_HOST)) {
- return;
- }
- if (! Helpers::validateUrl($id) || ! Helpers::validateUrl($actor)) {
- return;
- }
- if (! Helpers::validateLocalUrl($storyUrl)) {
- return;
- }
- if (! Helpers::validateLocalUrl($to)) {
- return;
- }
- if (Status::whereObjectUrl($id)->exists()) {
- return;
- }
- $storyId = Str::of($storyUrl)->explode('/')->last();
- $targetProfile = Helpers::profileFetch($to);
- $story = Story::whereProfileId($targetProfile->id)
- ->find($storyId);
- if (! $story) {
- return;
- }
- if ($story->can_react == false) {
- return;
- }
- $actorProfile = Helpers::profileFetch($actor);
- if (AccountService::blocksDomain($targetProfile->id, $actorProfile->domain) == true) {
- return;
- }
- if (! FollowerService::follows($actorProfile->id, $targetProfile->id)) {
- return;
- }
- $url = $id;
- if (str_ends_with($url, '/activity')) {
- $url = substr($url, 0, -9);
- }
- $status = new Status;
- $status->profile_id = $actorProfile->id;
- $status->type = 'story:reply';
- $status->caption = $text;
- $status->url = $url;
- $status->uri = $url;
- $status->object_url = $url;
- $status->scope = 'direct';
- $status->visibility = 'direct';
- $status->in_reply_to_profile_id = $story->profile_id;
- $status->entities = json_encode([
- 'story_id' => $story->id,
- 'caption' => $text,
- ]);
- $status->save();
- $dm = new DirectMessage;
- $dm->to_id = $story->profile_id;
- $dm->from_id = $actorProfile->id;
- $dm->type = 'story:comment';
- $dm->status_id = $status->id;
- $dm->meta = json_encode([
- 'story_username' => $targetProfile->username,
- 'story_actor_username' => $actorProfile->username,
- 'story_id' => $story->id,
- 'story_media_url' => url(Storage::url($story->path)),
- 'caption' => $text,
- ]);
- $dm->save();
- Conversation::updateOrInsert(
- [
- 'to_id' => $story->profile_id,
- 'from_id' => $actorProfile->id,
- ],
- [
- 'type' => 'story:comment',
- 'status_id' => $status->id,
- 'dm_id' => $dm->id,
- 'is_hidden' => false,
- ]
- );
- $n = new Notification;
- $n->profile_id = $dm->to_id;
- $n->actor_id = $dm->from_id;
- $n->item_id = $dm->id;
- $n->item_type = 'App\DirectMessage';
- $n->action = 'story:comment';
- $n->save();
- }
- public function handleFlagActivity()
- {
- if (! isset(
- $this->payload['id'],
- $this->payload['type'],
- $this->payload['actor'],
- $this->payload['object']
- )) {
- return;
- }
- $id = $this->payload['id'];
- $actor = $this->payload['actor'];
- if (Helpers::validateLocalUrl($id) || parse_url($id, PHP_URL_HOST) !== parse_url($actor, PHP_URL_HOST)) {
- return;
- }
- $content = null;
- if (isset($this->payload['content'])) {
- if (strlen($this->payload['content']) > 5000) {
- $content = app(SanitizeService::class)->html(substr($this->payload['content'], 0, 5000).' ... (truncated message due to exceeding max length)');
- } else {
- $content = app(SanitizeService::class)->html($this->payload['content']);
- }
- }
- $object = $this->payload['object'];
- if (empty($object) || (! is_array($object) && ! is_string($object))) {
- return;
- }
- if (is_array($object) && count($object) > 100) {
- return;
- }
- $objects = collect([]);
- $accountId = null;
- foreach ($object as $objectUrl) {
- if (! Helpers::validateLocalUrl($objectUrl)) {
- return;
- }
- if (str_contains($objectUrl, '/users/')) {
- $username = last(explode('/', $objectUrl));
- $profileId = Profile::whereUsername($username)->first();
- if ($profileId) {
- $accountId = $profileId->id;
- }
- } elseif (str_contains($objectUrl, '/p/')) {
- $postId = last(explode('/', $objectUrl));
- $objects->push($postId);
- } else {
- continue;
- }
- }
- if (! $accountId && ! $objects->count()) {
- return;
- }
- if ($objects->count()) {
- $obc = $objects->count();
- if ($obc > 25) {
- if ($obc > 30) {
- return;
- } else {
- $objLimit = $objects->take(20);
- $objects = collect($objLimit->all());
- $obc = $objects->count();
- }
- }
- $count = Status::whereProfileId($accountId)->find($objects)->count();
- if ($obc !== $count) {
- return;
- }
- }
- $instanceHost = parse_url($id, PHP_URL_HOST);
- $instance = Instance::updateOrCreate([
- 'domain' => $instanceHost,
- ]);
- $report = new RemoteReport;
- $report->status_ids = $objects->toArray();
- $report->comment = $content;
- $report->account_id = $accountId;
- $report->uri = $id;
- $report->instance_id = $instance->id;
- $report->report_meta = [
- 'actor' => $actor,
- 'object' => $object,
- ];
- $report->save();
- }
- public function handleUpdateActivity()
- {
- $activity = $this->payload['object'];
- if (! isset($activity['type'], $activity['id'])) {
- return;
- }
- if (! Helpers::validateUrl($activity['id'])) {
- return;
- }
- if ($activity['type'] === 'Note') {
- if (Status::whereObjectUrl($activity['id'])->exists()) {
- StatusRemoteUpdatePipeline::dispatch($activity);
- }
- } elseif ($activity['type'] === 'Person') {
- if (UpdatePersonValidator::validate($this->payload)) {
- HandleUpdateActivity::dispatch($this->payload)->onQueue('low');
- }
- }
- }
- public function handleMoveActivity()
- {
- $actor = $this->payload['actor'];
- $activity = $this->payload['object'];
- $target = $this->payload['target'];
- if (
- ! Helpers::validateUrl($actor) ||
- ! Helpers::validateUrl($activity) ||
- ! Helpers::validateUrl($target)
- ) {
- return;
- }
- Bus::chain([
- new ProcessMovePipeline($target, $activity),
- new MoveMigrateFollowersPipeline($target, $activity),
- new UnfollowLegacyAccountMovePipeline($target, $activity),
- new CleanupLegacyAccountMovePipeline($target, $activity),
- ])
- ->catch(function (Throwable $e) {
- Log::error($e);
- })
- ->onQueue('move')
- ->delay(now()->addMinutes(random_int(1, 3)))
- ->dispatch();
- }
- }
|