123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625 |
- <?php
- namespace App\Util\ActivityPub;
- use Cache, DB, Log, Purify, Redis, Validator;
- use App\{
- Activity,
- DirectMessage,
- Follower,
- FollowRequest,
- Like,
- Notification,
- Media,
- Profile,
- Status,
- StatusHashtag,
- UserFilter
- };
- use Carbon\Carbon;
- use App\Util\ActivityPub\Helpers;
- use Illuminate\Support\Str;
- use App\Jobs\LikePipeline\LikePipeline;
- use App\Jobs\FollowPipeline\FollowPipeline;
- use App\Jobs\DeletePipeline\DeleteRemoteProfilePipeline;
- use App\Util\ActivityPub\Validator\Accept as AcceptValidator;
- use App\Util\ActivityPub\Validator\Add as AddValidator;
- 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\UndoFollow as UndoFollowValidator;
- 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();
- // if(!Activity::where('data->id', $this->payload['id'])->exists()) {
- // (new Activity())->create([
- // 'to_id' => $this->profile->id,
- // 'data' => json_encode($this->payload)
- // ]);
- // }
- return;
- }
- public function handleVerb()
- {
- $verb = (string) $this->payload['type'];
- switch ($verb) {
- case 'Add':
- if(AddValidator::validate($this->payload) == false) { return; }
- $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':
- $this->handleRejectActivity();
- break;
- case 'Undo':
- $this->handleUndoActivity();
- 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 ;)
- }
- public function handleCreateActivity()
- {
- $activity = $this->payload['object'];
- $actor = $this->actorFirstOrCreate($this->payload['actor']);
- if(!$actor || $actor->domain == null) {
- return;
- }
- $to = $activity['to'];
- $cc = isset($activity['cc']) ? $activity['cc'] : [];
- if(count($to) == 1 &&
- count($cc) == 0 &&
- parse_url($to[0], PHP_URL_HOST) == config('pixelfed.domain.app')
- ) {
- $this->handleDirectMessage();
- return;
- }
- if(!$this->verifyNoteAttachment()) {
- return;
- }
- if($activity['type'] == 'Note' && !empty($activity['inReplyTo'])) {
- $this->handleNoteReply();
- } elseif($activity['type'] == 'Note' && !empty($activity['attachment'])) {
- $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);
- return;
- }
- public function handleNoteCreate()
- {
- $activity = $this->payload['object'];
- $actor = $this->actorFirstOrCreate($this->payload['actor']);
- if(!$actor || $actor->domain == null) {
- return;
- }
- if($actor->followers()->count() == 0) {
- return;
- }
- $url = isset($activity['url']) ? $activity['url'] : $activity['id'];
- if(Status::whereUrl($url)->exists()) {
- return;
- }
- Helpers::statusFetch($url);
- return;
- }
- 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(in_array($actor->id, $profile->blockedIds()->toArray())) {
- return;
- }
- $msg = $activity['content'];
- $msgText = strip_tags($activity['content']);
- 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->rendered = $msg;
- $status->visibility = 'direct';
- $status->scope = 'direct';
- $status->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();
- if(count($activity['attachment'])) {
- $photos = 0;
- $videos = 0;
- $allowed = explode(',', config('pixelfed.media_types'));
- $activity['attachment'] = array_slice($activity['attachment'], 0, config('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->message = $dm->toText();
- $notification->rendered = $dm->toHtml();
- $notification->item_id = $dm->id;
- $notification->item_type = "App\DirectMessage";
- $notification->save();
- }
- return;
- }
- public function handleFollowActivity()
- {
- $actor = $this->actorFirstOrCreate($this->payload['actor']);
- $target = $this->actorFirstOrCreate($this->payload['object']);
- if(!$actor || $actor->domain == null || $target->domain !== null) {
- return;
- }
- if(
- Follower::whereProfileId($actor->id)
- ->whereFollowingId($target->id)
- ->exists() ||
- FollowRequest::whereFollowerId($actor->id)
- ->whereFollowingId($target->id)
- ->exists()
- ) {
- return;
- }
- if($target->is_private == true) {
- FollowRequest::firstOrCreate([
- 'follower_id' => $actor->id,
- 'following_id' => $target->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);
- } 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);
- // 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);
- }
- }
- public function handleAnnounceActivity()
- {
- $actor = $this->actorFirstOrCreate($this->payload['actor']);
- $activity = $this->payload['object'];
- if(!$actor || $actor->domain == null) {
- return;
- }
- if(Helpers::validateLocalUrl($activity) == false) {
- return;
- }
- $parent = Helpers::statusFetch($activity);
- if(empty($parent)) {
- return;
- }
- $status = Status::firstOrCreate([
- 'profile_id' => $actor->id,
- 'reblog_of_id' => $parent->id,
- 'type' => 'share'
- ]);
- Notification::firstOrCreate([
- 'profile_id' => $parent->profile->id,
- 'actor_id' => $actor->id,
- 'action' => 'share',
- 'message' => $status->replyToText(),
- 'rendered' => $status->replyToHtml(),
- 'item_id' => $parent->id,
- 'item_type' => 'App\Status'
- ]);
- $parent->reblogs_count = $parent->shares()->count();
- $parent->save();
- }
- 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);
- $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);
- $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::dispatchNow($profile);
- return;
- } else {
- $type = $this->payload['object']['type'];
- $typeCheck = in_array($type, ['Person', 'Tombstone']);
- 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::dispatchNow($profile);
- return;
- break;
- case 'Tombstone':
- $profile = Helpers::profileFetch($actor);
- $status = Status::whereProfileId($profile->id)
- ->whereUri($id)
- ->orWhere('url', $id)
- ->orWhere('object_url', $id)
- ->first();
- if(!$status) {
- return;
- }
- $status->directMessage()->delete();
- $status->media()->delete();
- $status->likes()->delete();
- $status->shares()->delete();
- $status->delete();
- 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;
- }
- $like = Like::firstOrCreate([
- 'profile_id' => $profile->id,
- 'status_id' => $status->id
- ]);
- if($like->wasRecentlyCreated == true) {
- $status->likes_count = $status->likes()->count();
- $status->save();
- LikePipeline::dispatch($like);
- }
- return;
- }
- public function handleRejectActivity()
- {
- }
- public function handleUndoActivity()
- {
- $actor = $this->payload['actor'];
- $profile = self::actorFirstOrCreate($actor);
- $obj = $this->payload['object'];
- switch ($obj['type']) {
- case 'Accept':
- break;
-
- case 'Announce':
- $obj = $obj['object'];
- if(!Helpers::validateLocalUrl($obj)) {
- return;
- }
- $status = Helpers::statusFetch($obj);
- if(!$status) {
- return;
- }
- Status::whereProfileId($profile->id)
- ->whereReblogOfId($status->id)
- ->forceDelete();
- 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;
- }
- Follower::whereProfileId($profile->id)
- ->whereFollowingId($following->id)
- ->delete();
- Notification::whereProfileId($following->id)
- ->whereActorId($profile->id)
- ->whereAction('follow')
- ->whereItemId($following->id)
- ->whereItemType('App\Profile')
- ->forceDelete();
- break;
-
- case 'Like':
- $status = Helpers::statusFirstOrFetch($obj['object']);
- if(!$status) {
- 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;
- }
- return;
- }
- }
|