kopia lustrzana https://github.com/pixelfed/pixelfed
				
				
				
			
		
			
				
	
	
		
			330 wiersze
		
	
	
		
			9.8 KiB
		
	
	
	
		
			PHP
		
	
	
			
		
		
	
	
			330 wiersze
		
	
	
		
			9.8 KiB
		
	
	
	
		
			PHP
		
	
	
<?php
 | 
						|
 | 
						|
namespace App\Http\Controllers;
 | 
						|
 | 
						|
use App\EmailVerification;
 | 
						|
use App\Follower;
 | 
						|
use App\FollowRequest;
 | 
						|
use App\Jobs\FollowPipeline\FollowPipeline;
 | 
						|
use App\Mail\ConfirmEmail;
 | 
						|
use App\Notification;
 | 
						|
use App\Profile;
 | 
						|
use App\User;
 | 
						|
use App\UserFilter;
 | 
						|
use Auth;
 | 
						|
use Cache;
 | 
						|
use Carbon\Carbon;
 | 
						|
use Illuminate\Http\Request;
 | 
						|
use Mail;
 | 
						|
use Redis;
 | 
						|
use PragmaRX\Google2FA\Google2FA;
 | 
						|
 | 
						|
class AccountController extends Controller
 | 
						|
{
 | 
						|
    protected $filters = [
 | 
						|
      'user.mute',
 | 
						|
      'user.block',
 | 
						|
    ];
 | 
						|
 | 
						|
    public function __construct()
 | 
						|
    {
 | 
						|
        $this->middleware('auth');
 | 
						|
    }
 | 
						|
 | 
						|
    public function notifications(Request $request)
 | 
						|
    {
 | 
						|
        $this->validate($request, [
 | 
						|
          'page' => 'nullable|min:1|max:3',
 | 
						|
          'a'    => 'nullable|alpha_dash',
 | 
						|
      ]);
 | 
						|
        $profile = Auth::user()->profile;
 | 
						|
        $action = $request->input('a');
 | 
						|
        $timeago = Carbon::now()->subMonths(6);
 | 
						|
        if ($action && in_array($action, ['comment', 'follow', 'mention'])) {
 | 
						|
            $notifications = Notification::whereProfileId($profile->id)
 | 
						|
            ->whereAction($action)
 | 
						|
            ->whereDate('created_at', '>', $timeago)
 | 
						|
            ->orderBy('id', 'desc')
 | 
						|
            ->simplePaginate(30);
 | 
						|
        } else {
 | 
						|
            $notifications = Notification::whereProfileId($profile->id)
 | 
						|
            ->whereDate('created_at', '>', $timeago)
 | 
						|
            ->orderBy('id', 'desc')
 | 
						|
            ->simplePaginate(30);
 | 
						|
        }
 | 
						|
 | 
						|
        return view('account.activity', compact('profile', 'notifications'));
 | 
						|
    }
 | 
						|
 | 
						|
    public function followingActivity(Request $request)
 | 
						|
    {
 | 
						|
        $this->validate($request, [
 | 
						|
          'page' => 'nullable|min:1|max:3',
 | 
						|
          'a'    => 'nullable|alpha_dash',
 | 
						|
      ]);
 | 
						|
        $profile = Auth::user()->profile;
 | 
						|
        $action = $request->input('a');
 | 
						|
        $timeago = Carbon::now()->subMonths(3);
 | 
						|
        $following = $profile->following->pluck('id');
 | 
						|
        $notifications = Notification::whereIn('actor_id', $following)
 | 
						|
          ->where('profile_id', '!=', $profile->id)
 | 
						|
          ->whereDate('created_at', '>', $timeago)
 | 
						|
          ->orderBy('notifications.created_at', 'desc')
 | 
						|
          ->simplePaginate(30);
 | 
						|
 | 
						|
        return view('account.following', compact('profile', 'notifications'));
 | 
						|
    }
 | 
						|
 | 
						|
    public function verifyEmail(Request $request)
 | 
						|
    {
 | 
						|
        return view('account.verify_email');
 | 
						|
    }
 | 
						|
 | 
						|
    public function sendVerifyEmail(Request $request)
 | 
						|
    {
 | 
						|
        $timeLimit = Carbon::now()->subDays(1)->toDateTimeString();
 | 
						|
        $recentAttempt = EmailVerification::whereUserId(Auth::id())
 | 
						|
          ->where('created_at', '>', $timeLimit)->count();
 | 
						|
        $exists = EmailVerification::whereUserId(Auth::id())->count();
 | 
						|
 | 
						|
        if ($recentAttempt == 1 && $exists == 1) {
 | 
						|
            return redirect()->back()->with('error', 'A verification email has already been sent recently. Please check your email, or try again later.');
 | 
						|
        } elseif ($recentAttempt == 0 && $exists !== 0) {
 | 
						|
            // Delete old verification and send new one.
 | 
						|
            EmailVerification::whereUserId(Auth::id())->delete();
 | 
						|
        }
 | 
						|
 | 
						|
        $user = User::whereNull('email_verified_at')->find(Auth::id());
 | 
						|
        $utoken = hash('sha512', $user->id);
 | 
						|
        $rtoken = str_random(40);
 | 
						|
 | 
						|
        $verify = new EmailVerification();
 | 
						|
        $verify->user_id = $user->id;
 | 
						|
        $verify->email = $user->email;
 | 
						|
        $verify->user_token = $utoken;
 | 
						|
        $verify->random_token = $rtoken;
 | 
						|
        $verify->save();
 | 
						|
 | 
						|
        Mail::to($user->email)->send(new ConfirmEmail($verify));
 | 
						|
 | 
						|
        return redirect()->back()->with('status', 'Verification email sent!');
 | 
						|
    }
 | 
						|
 | 
						|
    public function confirmVerifyEmail(Request $request, $userToken, $randomToken)
 | 
						|
    {
 | 
						|
        $verify = EmailVerification::where('user_token', $userToken)
 | 
						|
          ->where('random_token', $randomToken)
 | 
						|
          ->firstOrFail();
 | 
						|
 | 
						|
        if (Auth::id() === $verify->user_id) {
 | 
						|
            $user = User::find(Auth::id());
 | 
						|
            $user->email_verified_at = Carbon::now();
 | 
						|
            $user->save();
 | 
						|
 | 
						|
            return redirect('/');
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    public function fetchNotifications($id)
 | 
						|
    {
 | 
						|
        $key = config('cache.prefix').":user.{$id}.notifications";
 | 
						|
        $redis = Redis::connection();
 | 
						|
        $notifications = $redis->lrange($key, 0, 30);
 | 
						|
        if (empty($notifications)) {
 | 
						|
            $notifications = Notification::whereProfileId($id)
 | 
						|
          ->orderBy('id', 'desc')->take(30)->get();
 | 
						|
        } else {
 | 
						|
            $notifications = $this->hydrateNotifications($notifications);
 | 
						|
        }
 | 
						|
 | 
						|
        return $notifications;
 | 
						|
    }
 | 
						|
 | 
						|
    public function hydrateNotifications($keys)
 | 
						|
    {
 | 
						|
        $prefix = 'notification.';
 | 
						|
        $notifications = collect([]);
 | 
						|
        foreach ($keys as $key) {
 | 
						|
            $notifications->push(Cache::get("{$prefix}{$key}"));
 | 
						|
        }
 | 
						|
 | 
						|
        return $notifications;
 | 
						|
    }
 | 
						|
 | 
						|
    public function messages()
 | 
						|
    {
 | 
						|
        return view('account.messages');
 | 
						|
    }
 | 
						|
 | 
						|
    public function showMessage(Request $request, $id)
 | 
						|
    {
 | 
						|
        return view('account.message');
 | 
						|
    }
 | 
						|
 | 
						|
    public function mute(Request $request)
 | 
						|
    {
 | 
						|
        $this->validate($request, [
 | 
						|
          'type' => 'required|string',
 | 
						|
          'item' => 'required|integer|min:1',
 | 
						|
        ]);
 | 
						|
 | 
						|
        $user = Auth::user()->profile;
 | 
						|
        $type = $request->input('type');
 | 
						|
        $item = $request->input('item');
 | 
						|
        $action = "{$type}.mute";
 | 
						|
 | 
						|
        if (!in_array($action, $this->filters)) {
 | 
						|
            return abort(406);
 | 
						|
        }
 | 
						|
        $filterable = [];
 | 
						|
        switch ($type) {
 | 
						|
          case 'user':
 | 
						|
            $profile = Profile::findOrFail($item);
 | 
						|
            if ($profile->id == $user->id) {
 | 
						|
                return abort(403);
 | 
						|
            }
 | 
						|
            $class = get_class($profile);
 | 
						|
            $filterable['id'] = $profile->id;
 | 
						|
            $filterable['type'] = $class;
 | 
						|
            break;
 | 
						|
 | 
						|
          default:
 | 
						|
            // code...
 | 
						|
            break;
 | 
						|
        }
 | 
						|
 | 
						|
        $filter = UserFilter::firstOrCreate([
 | 
						|
          'user_id'         => $user->id,
 | 
						|
          'filterable_id'   => $filterable['id'],
 | 
						|
          'filterable_type' => $filterable['type'],
 | 
						|
          'filter_type'     => 'mute',
 | 
						|
        ]);
 | 
						|
 | 
						|
        return redirect()->back();
 | 
						|
    }
 | 
						|
 | 
						|
    public function block(Request $request)
 | 
						|
    {
 | 
						|
        $this->validate($request, [
 | 
						|
          'type' => 'required|string',
 | 
						|
          'item' => 'required|integer|min:1',
 | 
						|
        ]);
 | 
						|
 | 
						|
        $user = Auth::user()->profile;
 | 
						|
        $type = $request->input('type');
 | 
						|
        $item = $request->input('item');
 | 
						|
        $action = "{$type}.block";
 | 
						|
        if (!in_array($action, $this->filters)) {
 | 
						|
            return abort(406);
 | 
						|
        }
 | 
						|
        $filterable = [];
 | 
						|
        switch ($type) {
 | 
						|
          case 'user':
 | 
						|
            $profile = Profile::findOrFail($item);
 | 
						|
            $class = get_class($profile);
 | 
						|
            $filterable['id'] = $profile->id;
 | 
						|
            $filterable['type'] = $class;
 | 
						|
            break;
 | 
						|
 | 
						|
          default:
 | 
						|
            // code...
 | 
						|
            break;
 | 
						|
        }
 | 
						|
 | 
						|
        $filter = UserFilter::firstOrCreate([
 | 
						|
          'user_id'         => $user->id,
 | 
						|
          'filterable_id'   => $filterable['id'],
 | 
						|
          'filterable_type' => $filterable['type'],
 | 
						|
          'filter_type'     => 'block',
 | 
						|
        ]);
 | 
						|
 | 
						|
        return redirect()->back();
 | 
						|
    }
 | 
						|
 | 
						|
    public function followRequests(Request $request)
 | 
						|
    {
 | 
						|
        $pid = Auth::user()->profile->id;
 | 
						|
        $followers = FollowRequest::whereFollowingId($pid)->orderBy('id','desc')->whereIsRejected(0)->simplePaginate(10);
 | 
						|
        return view('account.follow-requests', compact('followers'));
 | 
						|
    }
 | 
						|
 | 
						|
    public function followRequestHandle(Request $request)
 | 
						|
    {
 | 
						|
        $this->validate($request, [
 | 
						|
            'action' => 'required|string|max:10',
 | 
						|
            'id' => 'required|integer|min:1'
 | 
						|
        ]);
 | 
						|
 | 
						|
        $pid = Auth::user()->profile->id;
 | 
						|
        $action = $request->input('action') === 'accept' ? 'accept' : 'reject';
 | 
						|
        $id = $request->input('id');
 | 
						|
        $followRequest = FollowRequest::whereFollowingId($pid)->findOrFail($id);
 | 
						|
        $follower = $followRequest->follower;
 | 
						|
 | 
						|
        switch ($action) {
 | 
						|
            case 'accept':
 | 
						|
                $follow = new Follower();
 | 
						|
                $follow->profile_id = $follower->id;
 | 
						|
                $follow->following_id = $pid;
 | 
						|
                $follow->save();
 | 
						|
                FollowPipeline::dispatch($follow);
 | 
						|
                $followRequest->delete();
 | 
						|
                break;
 | 
						|
 | 
						|
            case 'reject':
 | 
						|
                $followRequest->is_rejected = true;
 | 
						|
                $followRequest->save();
 | 
						|
                break;
 | 
						|
        }
 | 
						|
 | 
						|
        return response()->json(['msg' => 'success'], 200);
 | 
						|
    }
 | 
						|
 | 
						|
    public function sudoMode(Request $request)
 | 
						|
    {
 | 
						|
        return view('auth.sudo');
 | 
						|
    }
 | 
						|
 | 
						|
    public function sudoModeVerify(Request $request)
 | 
						|
    {
 | 
						|
        $this->validate($request, [
 | 
						|
            'password' => 'required|string|max:500'
 | 
						|
        ]);
 | 
						|
        $user = Auth::user();
 | 
						|
        $password = $request->input('password');
 | 
						|
        $next = $request->session()->get('redirectNext', '/');
 | 
						|
        if(password_verify($password, $user->password) === true) {
 | 
						|
            $request->session()->put('sudoMode', time());
 | 
						|
            return redirect($next);
 | 
						|
        } else {
 | 
						|
            return redirect()
 | 
						|
                ->back()
 | 
						|
                ->withErrors(['password' => __('auth.failed')]);
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    public function twoFactorCheckpoint(Request $request)
 | 
						|
    {
 | 
						|
        return view('auth.checkpoint');
 | 
						|
    }
 | 
						|
 | 
						|
    public function twoFactorVerify(Request $request)
 | 
						|
    {
 | 
						|
        $this->validate($request, [
 | 
						|
            'code'  => 'required|string|max:32'
 | 
						|
        ]);
 | 
						|
        $user = Auth::user();
 | 
						|
        $code = $request->input('code');
 | 
						|
        $google2fa = new Google2FA();
 | 
						|
        $verify = $google2fa->verifyKey($user->{'2fa_secret'}, $code);
 | 
						|
        if($verify) {
 | 
						|
            $request->session()->push('2fa.session.active', true);
 | 
						|
            return redirect('/');
 | 
						|
        } else {
 | 
						|
            return redirect()->back()->withErrors([
 | 
						|
                'code' => 'Invalid code'
 | 
						|
            ]);
 | 
						|
        }
 | 
						|
    }
 | 
						|
}
 |