<?php

namespace App\Http\Controllers;

use App\Jobs\InboxPipeline\InboxWorker;
use App\Jobs\RemoteFollowPipeline\RemoteFollowPipeline;
use App\Profile;
use App\Transformer\ActivityPub\ProfileOutbox;
use App\Util\Lexer\Nickname;
use App\Util\Webfinger\Webfinger;
use Auth;
use Cache;
use Carbon\Carbon;
use Illuminate\Http\Request;
use League\Fractal;

class FederationController extends Controller
{
    public function authCheck()
    {
        if (!Auth::check()) {
            return abort(403);
        }
    }

    public function authorizeFollow(Request $request)
    {
        $this->authCheck();
        $this->validate($request, [
        'acct' => 'required|string|min:3|max:255',
      ]);
        $acct = $request->input('acct');
        $nickname = Nickname::normalizeProfileUrl($acct);

        return view('federation.authorizefollow', compact('acct', 'nickname'));
    }

    public function remoteFollow()
    {
        $this->authCheck();

        return view('federation.remotefollow');
    }

    public function remoteFollowStore(Request $request)
    {
        $this->authCheck();
        $this->validate($request, [
        'url' => 'required|string',
      ]);

        if (config('pixelfed.remote_follow_enabled') !== true) {
            abort(403);
        }

        $follower = Auth::user()->profile;
        $url = $request->input('url');

        RemoteFollowPipeline::dispatch($follower, $url);

        return redirect()->back();
    }

    public function nodeinfoWellKnown()
    {
        $res = [
        'links' => [
          [
            'href' => config('pixelfed.nodeinfo.url'),
            'rel'  => 'http://nodeinfo.diaspora.software/ns/schema/2.0',
          ],
        ],
      ];

        return response()->json($res);
    }

    public function nodeinfo()
    {
        $res = Cache::remember('api:nodeinfo', 60, function () {
            return [
          'metadata' => [
            'nodeName' => config('app.name'),
            'software' => [
              'homepage' => 'https://pixelfed.org',
              'github'   => 'https://github.com/pixelfed',
              'follow'   => 'https://mastodon.social/@pixelfed',
            ],
          ],
          'openRegistrations' => config('pixelfed.open_registration'),
          'protocols'         => [
            'activitypub',
          ],
          'services' => [
            'inbound'  => [],
            'outbound' => [],
          ],
          'software' => [
            'name'    => 'pixelfed',
            'version' => config('pixelfed.version'),
          ],
          'usage' => [
            'localPosts'    => \App\Status::whereLocal(true)->whereHas('media')->count(),
            'localComments' => \App\Status::whereLocal(true)->whereNotNull('in_reply_to_id')->count(),
            'users'         => [
              'total'          => \App\User::count(),
              'activeHalfyear' => \App\User::where('updated_at', '>', Carbon::now()->subMonths(6)->toDateTimeString())->count(),
              'activeMonth'    => \App\User::where('updated_at', '>', Carbon::now()->subMonths(1)->toDateTimeString())->count(),
            ],
          ],
          'version' => '2.0',
        ];
        });

        return response()->json($res, 200, [], JSON_PRETTY_PRINT);
    }

    public function webfinger(Request $request)
    {
        $this->validate($request, ['resource'=>'required|string|min:3|max:255']);

        $hash = hash('sha256', $request->input('resource'));

        $webfinger = Cache::remember('api:webfinger:'.$hash, 1440, function () use ($request) {
            $resource = $request->input('resource');
            $parsed = Nickname::normalizeProfileUrl($resource);
            $username = $parsed['username'];
            $user = Profile::whereUsername($username)->firstOrFail();

            return (new Webfinger($user))->generate();
        });

        return response()->json($webfinger, 200, [], JSON_PRETTY_PRINT);
    }

    public function userOutbox(Request $request, $username)
    {
        if (config('pixelfed.activitypub_enabled') == false) {
            abort(403);
        }

        $user = Profile::whereNull('remote_url')->whereUsername($username)->firstOrFail();
        $timeline = $user->statuses()->orderBy('created_at', 'desc')->paginate(10);
        $fractal = new Fractal\Manager();
        $resource = new Fractal\Resource\Item($user, new ProfileOutbox());
        $res = $fractal->createData($resource)->toArray();

        return response(json_encode($res['data']))->header('Content-Type', 'application/activity+json');
    }

    public function userInbox(Request $request, $username)
    {
        if (config('pixelfed.activitypub_enabled') == false) {
            abort(403);
        }
        $mimes = [
        'application/activity+json',
        'application/ld+json; profile="https://www.w3.org/ns/activitystreams"',
      ];
        if (!in_array($request->header('Content-Type'), $mimes)) {
            abort(500, 'Invalid request');
        }
        $profile = Profile::whereUsername($username)->firstOrFail();
        InboxWorker::dispatch($request, $profile, $request->all());
    }
}