@ -2,442 +2,424 @@
namespace App\Http\Controllers;
namespace App\Http\Controllers;
use Illuminate\Http\Request;
use App\AccountInterstitial;
use App\{
use App\Bookmark;
AccountInterstitial,
use App\DirectMessage;
Bookmark,
use App\DiscoverCategory;
DirectMessage,
use App\Follower;
DiscoverCategory,
Hashtag,
Follower,
Like,
Media,
MediaTag,
Notification,
Profile,
StatusHashtag,
Status,
User,
UserFilter,
};
use Auth,Cache;
use Illuminate\Support\Facades\Redis;
use Carbon\Carbon;
use League\Fractal;
use App\Transformer\Api\{
AccountTransformer,
StatusTransformer,
// StatusMediaContainerTransformer,
};
use App\Util\Media\Filter;
use App\Jobs\StatusPipeline\NewStatusPipeline;
use App\Jobs\ModPipeline\HandleSpammerPipeline;
use App\Jobs\ModPipeline\HandleSpammerPipeline;
use League\Fractal\Serializer\ArraySerializer;
use App\Profile;
use League\Fractal\Pagination\IlluminatePaginatorAdapter;
use App\Services\BookmarkService;
use Illuminate\Validation\Rule;
use App\Services\DiscoverService;
use Illuminate\Support\Str;
use App\Services\MediaTagService;
use App\Services\ModLogService;
use App\Services\ModLogService;
use App\Services\PublicTimelineService;
use App\Services\PublicTimelineService;
use App\Services\SnowflakeService;
use App\Services\StatusService;
use App\Services\StatusService;
use App\Services\UserFilterService;
use App\Services\UserFilterService;
use App\Services\DiscoverService;
use App\Status; // StatusMediaContainerTransformer,
use App\Services\BookmarkService;
use App\Transformer\Api\StatusTransformer;
use App\User;
use Auth;
use Cache;
use Illuminate\Http\Request;
use Illuminate\Support\Facades\Redis;
use Illuminate\Validation\Rule;
use League\Fractal;
use League\Fractal\Serializer\ArraySerializer;
class InternalApiController extends Controller
class InternalApiController extends Controller
{
{
protected $fractal;
protected $fractal;
public function __construct()
public function __construct()
{
{
$this->middleware('auth');
$this->middleware('auth');
$this->fractal = new Fractal\Manager();
$this->fractal = new Fractal\Manager;
$this->fractal->setSerializer(new ArraySerializer());
$this->fractal->setSerializer(new ArraySerializer);
}
}
// deprecated v2 compose api
// deprecated v2 compose api
public function compose(Request $request)
public function compose(Request $request)
{
{
return redirect('/');
return redirect('/');
}
}
// deprecated
// deprecated
public function discover(Request $request)
public function discover(Request $request) {}
{
return;
public function discoverPosts(Request $request)
}
{
$pid = $request->user()->profile_id;
public function discoverPosts(Request $request)
$filters = UserFilterService::filters($pid);
{
$forYou = DiscoverService::getForYou();
$pid = $request->user()->profile_id;
$posts = $forYou->take(50)->map(function ($post) {
$filters = UserFilterService::filters($pid);
return StatusService::get($post);
$forYou = DiscoverService::getForYou();
})
$posts = $forYou->take(50)->map(function($post) {
->filter(function ($post) use ($filters) {
return StatusService::get($post);
return $post & &
})
isset($post['account']) & &
->filter(function($post) use($filters) {
isset($post['account']['id']) & &
return $post & &
! in_array($post['account']['id'], $filters);
isset($post['account']) & &
})
isset($post['account']['id']) & &
->take(12)
!in_array($post['account']['id'], $filters);
->values();
})
->take(12)
return response()->json(compact('posts'));
->values();
}
return response()->json(compact('posts'));
}
public function directMessage(Request $request, $profileId, $threadId)
{
public function directMessage(Request $request, $profileId, $threadId)
$profile = Auth::user()->profile;
{
$profile = Auth::user()->profile;
if ($profileId != $profile->id) {
abort(403);
if($profileId != $profile->id) {
}
abort(403);
}
$msg = DirectMessage::whereToId($profile->id)
->orWhere('from_id', $profile->id)
$msg = DirectMessage::whereToId($profile->id)
->findOrFail($threadId);
->orWhere('from_id',$profile->id)
->findOrFail($threadId);
$thread = DirectMessage::with('status')->whereIn('to_id', [$profile->id, $msg->from_id])
->whereIn('from_id', [$profile->id, $msg->from_id])
$thread = DirectMessage::with('status')->whereIn('to_id', [$profile->id, $msg->from_id])
->orderBy('created_at', 'asc')
->whereIn('from_id', [$profile->id,$msg->from_id])
->paginate(30);
->orderBy('created_at', 'asc')
->paginate(30);
return response()->json(compact('msg', 'profile', 'thread'), 200, [], JSON_PRETTY_PRINT);
}
return response()->json(compact('msg', 'profile', 'thread'), 200, [], JSON_PRETTY_PRINT);
}
public function statusReplies(Request $request, int $id)
{
public function statusReplies(Request $request, int $id)
$this->validate($request, [
{
'limit' => 'nullable|int|min:1|max:6',
$this->validate($request, [
]);
'limit' => 'nullable|int|min:1|max:6'
$parent = Status::whereScope('public')->findOrFail($id);
]);
$limit = $request->input('limit') ?? 3;
$parent = Status::whereScope('public')->findOrFail($id);
$children = Status::whereInReplyToId($parent->id)
$limit = $request->input('limit') ?? 3;
->orderBy('created_at', 'desc')
$children = Status::whereInReplyToId($parent->id)
->take($limit)
->orderBy('created_at', 'desc')
->get();
->take($limit)
$resource = new Fractal\Resource\Collection($children, new StatusTransformer);
->get();
$res = $this->fractal->createData($resource)->toArray();
$resource = new Fractal\Resource\Collection($children, new StatusTransformer());
$res = $this->fractal->createData($resource)->toArray();
return response()->json($res);
}
return response()->json($res);
}
public function stories(Request $request) {}
public function stories(Request $request)
public function discoverCategories(Request $request)
{
{
$categories = DiscoverCategory::whereActive(true)->orderBy('order')->take(10)->get();
}
$res = $categories->map(function ($item) {
return [
public function discoverCategories(Request $request)
'name' => $item->name,
{
'url' => $item->url(),
$categories = DiscoverCategory::whereActive(true)->orderBy('order')->take(10)->get();
'thumb' => $item->thumb(),
$res = $categories->map(function($item) {
];
return [
});
'name' => $item->name,
'url' => $item->url(),
return response()->json($res);
'thumb' => $item->thumb()
}
];
});
public function modAction(Request $request)
return response()->json($res);
{
}
abort_unless(Auth::user()->is_admin, 400);
$this->validate($request, [
public function modAction(Request $request)
'action' => [
{
'required',
abort_unless(Auth::user()->is_admin, 400);
'string',
$this->validate($request, [
Rule::in([
'action' => [
'addcw',
'required',
'remcw',
'string',
'unlist',
Rule::in([
'spammer',
'addcw',
]),
'remcw',
],
'unlist',
'item_id' => 'required|integer|min:1',
'spammer'
'item_type' => [
])
'required',
],
'string',
'item_id' => 'required|integer|min:1',
Rule::in(['profile', 'status']),
'item_type' => [
],
'required',
]);
'string',
Rule::in(['profile', 'status'])
$action = $request->input('action');
]
$item_id = $request->input('item_id');
]);
$item_type = $request->input('item_type');
$action = $request->input('action');
$status = Status::findOrFail($item_id);
$item_id = $request->input('item_id');
$author = User::whereProfileId($status->profile_id)->first();
$item_type = $request->input('item_type');
abort_if($author & & $author->is_admin, 422, 'Cannot moderate administrator accounts');
$status = Status::findOrFail($item_id);
switch ($action) {
$author = User::whereProfileId($status->profile_id)->first();
case 'addcw':
abort_if($author & & $author->is_admin, 422, 'Cannot moderate administrator accounts');
$status->is_nsfw = true;
$status->save();
switch($action) {
ModLogService::boot()
case 'addcw':
->user(Auth::user())
$status->is_nsfw = true;
->objectUid($status->profile->user_id)
$status->save();
->objectId($status->id)
ModLogService::boot()
->objectType('App\Status::class')
->user(Auth::user())
->action('admin.status.moderate')
->objectUid($status->profile->user_id)
->metadata([
->objectId($status->id)
'action' => 'cw',
->objectType('App\Status::class')
'message' => 'Success!',
->action('admin.status.moderate')
])
->metadata([
->accessLevel('admin')
'action' => 'cw',
->save();
'message' => 'Success!'
])
if ($status->uri == null) {
->accessLevel('admin')
$media = $status->media;
->save();
$ai = new AccountInterstitial;
$ai->user_id = $status->profile->user_id;
if($status->uri == null) {
$ai->type = 'post.cw';
$media = $status->media;
$ai->view = 'account.moderation.post.cw';
$ai = new AccountInterstitial;
$ai->item_type = 'App\Status';
$ai->user_id = $status->profile->user_id;
$ai->item_id = $status->id;
$ai->type = 'post.cw';
$ai->has_media = (bool) $media->count();
$ai->view = 'account.moderation.post.cw';
$ai->blurhash = $media->count() ? $media->first()->blurhash : null;
$ai->item_type = 'App\Status';
$ai->meta = json_encode([
$ai->item_id = $status->id;
'caption' => $status->caption,
$ai->has_media = (bool) $media->count();
'created_at' => $status->created_at,
$ai->blurhash = $media->count() ? $media->first()->blurhash : null;
'type' => $status->type,
$ai->meta = json_encode([
'url' => $status->url(),
'caption' => $status->caption,
'is_nsfw' => $status->is_nsfw,
'created_at' => $status->created_at,
'scope' => $status->scope,
'type' => $status->type,
'reblog' => $status->reblog_of_id,
'url' => $status->url(),
'likes_count' => $status->likes_count,
'is_nsfw' => $status->is_nsfw,
'reblogs_count' => $status->reblogs_count,
'scope' => $status->scope,
]);
'reblog' => $status->reblog_of_id,
$ai->save();
'likes_count' => $status->likes_count,
'reblogs_count' => $status->reblogs_count,
$u = $status->profile->user;
]);
$u->has_interstitial = true;
$ai->save();
$u->save();
}
$u = $status->profile->user;
break;
$u->has_interstitial = true;
$u->save();
case 'remcw':
}
$status->is_nsfw = false;
break;
$status->save();
ModLogService::boot()
case 'remcw':
->user(Auth::user())
$status->is_nsfw = false;
->objectUid($status->profile->user_id)
$status->save();
->objectId($status->id)
ModLogService::boot()
->objectType('App\Status::class')
->user(Auth::user())
->action('admin.status.moderate')
->objectUid($status->profile->user_id)
->metadata([
->objectId($status->id)
'action' => 'remove_cw',
->objectType('App\Status::class')
'message' => 'Success!',
->action('admin.status.moderate')
])
->metadata([
->accessLevel('admin')
'action' => 'remove_cw',
->save();
'message' => 'Success!'
if ($status->uri == null) {
])
$ai = AccountInterstitial::whereUserId($status->profile->user_id)
->accessLevel('admin')
->whereType('post.cw')
->save();
->whereItemId($status->id)
if($status->uri == null) {
->whereItemType('App\Status')
$ai = AccountInterstitial::whereUserId($status->profile->user_id)
->first();
->whereType('post.cw')
$ai->delete();
->whereItemId($status->id)
}
->whereItemType('App\Status')
break;
->first();
$ai->delete();
case 'unlist':
}
$status->scope = $status->visibility = 'unlisted';
break;
$status->save();
PublicTimelineService::del($status->id);
case 'unlist':
ModLogService::boot()
$status->scope = $status->visibility = 'unlisted';
->user(Auth::user())
$status->save();
->objectUid($status->profile->user_id)
PublicTimelineService::del($status->id);
->objectId($status->id)
ModLogService::boot()
->objectType('App\Status::class')
->user(Auth::user())
->action('admin.status.moderate')
->objectUid($status->profile->user_id)
->metadata([
->objectId($status->id)
'action' => 'unlist',
->objectType('App\Status::class')
'message' => 'Success!',
->action('admin.status.moderate')
])
->metadata([
->accessLevel('admin')
'action' => 'unlist',
->save();
'message' => 'Success!'
])
if ($status->uri == null) {
->accessLevel('admin')
$media = $status->media;
->save();
$ai = new AccountInterstitial;
$ai->user_id = $status->profile->user_id;
if($status->uri == null) {
$ai->type = 'post.unlist';
$media = $status->media;
$ai->view = 'account.moderation.post.unlist';
$ai = new AccountInterstitial;
$ai->item_type = 'App\Status';
$ai->user_id = $status->profile->user_id;
$ai->item_id = $status->id;
$ai->type = 'post.unlist';
$ai->has_media = (bool) $media->count();
$ai->view = 'account.moderation.post.unlist';
$ai->blurhash = $media->count() ? $media->first()->blurhash : null;
$ai->item_type = 'App\Status';
$ai->meta = json_encode([
$ai->item_id = $status->id;
'caption' => $status->caption,
$ai->has_media = (bool) $media->count();
'created_at' => $status->created_at,
$ai->blurhash = $media->count() ? $media->first()->blurhash : null;
'type' => $status->type,
$ai->meta = json_encode([
'url' => $status->url(),
'caption' => $status->caption,
'is_nsfw' => $status->is_nsfw,
'created_at' => $status->created_at,
'scope' => $status->scope,
'type' => $status->type,
'reblog' => $status->reblog_of_id,
'url' => $status->url(),
'likes_count' => $status->likes_count,
'is_nsfw' => $status->is_nsfw,
'reblogs_count' => $status->reblogs_count,
'scope' => $status->scope,
]);
'reblog' => $status->reblog_of_id,
$ai->save();
'likes_count' => $status->likes_count,
'reblogs_count' => $status->reblogs_count,
$u = $status->profile->user;
]);
$u->has_interstitial = true;
$ai->save();
$u->save();
}
$u = $status->profile->user;
break;
$u->has_interstitial = true;
$u->save();
case 'spammer':
}
HandleSpammerPipeline::dispatch($status->profile);
break;
ModLogService::boot()
->user(Auth::user())
case 'spammer':
->objectUid($status->profile->user_id)
HandleSpammerPipeline::dispatch($status->profile);
->objectId($status->id)
ModLogService::boot()
->objectType('App\User::class')
->user(Auth::user())
->action('admin.status.moderate')
->objectUid($status->profile->user_id)
->metadata([
->objectId($status->id)
'action' => 'spammer',
->objectType('App\User::class')
'message' => 'Success!',
->action('admin.status.moderate')
])
->metadata([
->accessLevel('admin')
'action' => 'spammer',
->save();
'message' => 'Success!'
break;
])
}
->accessLevel('admin')
->save();
StatusService::del($status->id, true);
break;
}
return ['msg' => 200];
}
StatusService::del($status->id, true);
return ['msg' => 200];
public function composePost(Request $request)
}
{
abort(400, 'Endpoint deprecated');
public function composePost(Request $request)
}
{
abort(400, 'Endpoint deprecated');
public function bookmarks(Request $request)
}
{
$pid = $request->user()->profile_id;
public function bookmarks(Request $request)
$res = Bookmark::whereProfileId($pid)
{
->orderByDesc('created_at')
$pid = $request->user()->profile_id;
->simplePaginate(10)
$res = Bookmark::whereProfileId($pid)
->map(function ($bookmark) use ($pid) {
->orderByDesc('created_at')
$status = StatusService::get($bookmark->status_id, false);
->simplePaginate(10)
if (! $status) {
->map(function($bookmark) use($pid) {
return false;
$status = StatusService::get($bookmark->status_id, false);
}
if(!$status) {
$status['bookmarked_at'] = str_replace('+00:00', 'Z', $bookmark->created_at->format(DATE_RFC3339_EXTENDED));
return false;
}
if ($status) {
$status['bookmarked_at'] = str_replace('+00:00', 'Z', $bookmark->created_at->format(DATE_RFC3339_EXTENDED));
BookmarkService::add($pid, $status['id']);
}
if($status) {
BookmarkService::add($pid, $status['id']);
return $status;
}
})
return $status;
->filter(function ($bookmark) {
})
return $bookmark & & isset($bookmark['id']);
->filter(function($bookmark) {
})
return $bookmark & & isset($bookmark['id']);
->values();
})
->values();
return response()->json($res);
}
return response()->json($res);
}
public function accountStatuses(Request $request, $id)
{
public function accountStatuses(Request $request, $id)
$this->validate($request, [
{
'only_media' => 'nullable',
$this->validate($request, [
'pinned' => 'nullable',
'only_media' => 'nullable',
'exclude_replies' => 'nullable',
'pinned' => 'nullable',
'max_id' => 'nullable|integer|min:0|max:'.PHP_INT_MAX,
'exclude_replies' => 'nullable',
'since_id' => 'nullable|integer|min:0|max:'.PHP_INT_MAX,
'max_id' => 'nullable|integer|min:0|max:' . PHP_INT_MAX,
'min_id' => 'nullable|integer|min:0|max:'.PHP_INT_MAX,
'since_id' => 'nullable|integer|min:0|max:' . PHP_INT_MAX,
'limit' => 'nullable|integer|min:1|max:24',
'min_id' => 'nullable|integer|min:0|max:' . PHP_INT_MAX,
]);
'limit' => 'nullable|integer|min:1|max:24'
]);
$profile = Profile::whereNull('status')->findOrFail($id);
$profile = Profile::whereNull('status')->findOrFail($id);
$limit = $request->limit ?? 9;
$max_id = $request->max_id;
$limit = $request->limit ?? 9;
$min_id = $request->min_id;
$max_id = $request->max_id;
$scope = $request->only_media == true ?
$min_id = $request->min_id;
['photo', 'photo:album', 'video', 'video:album'] :
$scope = $request->only_media == true ?
['photo', 'photo:album', 'video', 'video:album', 'share', 'reply'];
['photo', 'photo:album', 'video', 'video:album'] :
['photo', 'photo:album', 'video', 'video:album', 'share', 'reply'];
if ($profile->is_private) {
if (! Auth::check()) {
if($profile->is_private) {
return response()->json([]);
if(!Auth::check()) {
}
return response()->json([]);
$pid = Auth::user()->profile->id;
}
$following = Cache::remember('profile:following:'.$pid, now()->addMinutes(1440), function () use ($pid) {
$pid = Auth::user()->profile->id;
$following = Follower::whereProfileId($pid)->pluck('following_id');
$following = Cache::remember('profile:following:'.$pid, now()->addMinutes(1440), function() use($pid) {
$following = Follower::whereProfileId($pid)->pluck('following_id');
return $following->push($pid)->toArray();
return $following->push($pid)->toArray();
});
});
$visibility = in_array($profile->id, $following) == true ? ['public', 'unlisted', 'private'] : [];
$visibility = true == in_array($profile->id, $following) ? ['public', 'unlisted', 'private'] : [];
} else {
} else {
if (Auth::check()) {
if(Auth::check()) {
$pid = Auth::user()->profile->id;
$pid = Auth::user()->profile->id;
$following = Cache::remember('profile:following:'.$pid, now()->addMinutes(1440), function () use ($pid) {
$following = Cache::remember('profile:following:'.$pid, now()->addMinutes(1440), function() use($pid) {
$following = Follower::whereProfileId($pid)->pluck('following_id');
$following = Follower::whereProfileId($pid)->pluck('following_id');
return $following->push($pid)->toArray();
return $following->push($pid)->toArray();
});
});
$visibility = true == in_array($profile->id, $following) ? ['public', 'unlisted', 'private'] : ['public', 'unlisted'];
$visibility = in_array($profile->id, $following) == true ? ['public', 'unlisted', 'private'] : ['public', 'unlisted'];
} else {
} else {
$visibility = ['public', 'unlisted'];
$visibility = ['public', 'unlisted'];
}
}
}
}
$dir = $min_id ? '>' : '< ';
$dir = $min_id ? '>' : '< ';
$id = $min_id ?? $max_id;
$id = $min_id ?? $max_id;
$timeline = Status::select(
$timeline = Status::select(
'id',
'id',
'uri',
'uri',
'caption',
'caption',
'rendered',
'profile_id',
'profile_id',
'type',
'type',
'in_reply_to_id',
'in_reply_to_id',
'reblog_of_id',
'reblog_of_id',
'is_nsfw',
'is_nsfw',
'likes_count',
'likes_count',
'reblogs_count',
'reblogs_count',
'scope',
'scope',
'local',
'local',
'created_at',
'created_at',
'updated_at'
'updated_at'
)->whereProfileId($profile->id)
)->whereProfileId($profile->id)
->whereIn('type', $scope)
->whereIn('type', $scope)
->where('id', $dir, $id)
->where('id', $dir, $id)
->whereIn('visibility', $visibility)
->whereIn('visibility', $visibility)
->latest()
->latest()
->limit($limit)
->limit($limit)
->get();
->get();
$resource = new Fractal\Resource\Collection($timeline, new StatusTransformer);
$resource = new Fractal\Resource\Collection($timeline, new StatusTransformer());
$res = $this->fractal->createData($resource)->toArray();
$res = $this->fractal->createData($resource)->toArray();
return response()->json($res);
return response()->json($res);
}
}
public function remoteProfile(Request $request, $id)
public function remoteProfile(Request $request, $id)
{
{
return redirect('/i/web/profile/'.$id);
return redirect('/i/web/profile/' . $id);
}
}
public function remoteStatus(Request $request, $profileId, $statusId)
public function remoteStatus(Request $request, $profileId, $statusId)
{
{
return redirect('/i/web/post/'.$statusId);
return redirect('/i/web/post/' . $statusId);
}
}
public function requestEmailVerification(Request $request)
public function requestEmailVerification(Request $request)
{
{
$pid = $request->user()->profile_id;
$pid = $request->user()->profile_id;
$exists = Redis::sismember('email:manual', $pid);
$exists = Redis::sismember('email:manual', $pid);
return view('account.email.request_verification', compact('exists'));
return view('account.email.request_verification', compact('exists'));
}
}
public function requestEmailVerificationStore(Request $request)
public function requestEmailVerificationStore(Request $request)
{
{
$pid = $request->user()->profile_id;
$pid = $request->user()->profile_id;
Redis::sadd('email:manual', $pid);
Redis::sadd('email:manual', $pid);
return redirect('/i/verify-email')->with(['status' => 'Successfully sent manual verification request!']);
}
return redirect('/i/verify-email')->with(['status' => 'Successfully sent manual verification request!']);
}
}
}