<?php

namespace App\Jobs\StoryPipeline;

use Illuminate\Bus\Queueable;
use Illuminate\Contracts\Queue\ShouldBeUnique;
use Illuminate\Contracts\Queue\ShouldQueue;
use Illuminate\Foundation\Bus\Dispatchable;
use Illuminate\Queue\InteractsWithQueue;
use Illuminate\Queue\SerializesModels;
use Storage;
use App\Story;
use League\Fractal;
use League\Fractal\Serializer\ArraySerializer;
use App\Transformer\ActivityPub\Verb\DeleteStory;
use App\Util\ActivityPub\Helpers;
use GuzzleHttp\Pool;
use GuzzleHttp\Client;
use GuzzleHttp\Promise;
use App\Util\ActivityPub\HttpSignature;
use App\Services\FollowerService;
use App\Services\StoryService;

class StoryExpire implements ShouldQueue
{
	use Dispatchable, InteractsWithQueue, Queueable, SerializesModels;

	protected $story;

	/**
	 * Delete the job if its models no longer exist.
	 *
	 * @var bool
	 */
	public $deleteWhenMissingModels = true;

	/**
	 * Create a new job instance.
	 *
	 * @return void
	 */
	public function __construct(Story $story)
	{
		$this->story = $story;
	}

	/**
	 * Execute the job.
	 *
	 * @return void
	 */
	public function handle()
	{
		$story = $this->story;

		if($story->local == false) {
			$this->handleRemoteExpiry();
			return;
		}

		if($story->active == false) {
			return;
		}

		if($story->expires_at->gt(now())) {
			return;
		}

		$story->active = false;
		$story->save();

		$this->rotateMediaPath();
		$this->fanoutExpiry();

		StoryService::delLatest($story->profile_id);
	}

	protected function rotateMediaPath()
	{
		$story = $this->story;
		$date = date('Y').date('m');
		$old = $story->path;
		$base = "story_archives/{$story->profile_id}/{$date}/";
		$paths = explode('/', $old);
		$path = array_pop($paths);
		$newPath = $base . $path;

		if(Storage::exists($old) == true) {
			$dir = implode('/', $paths);
			Storage::move($old, $newPath);
			Storage::delete($old);
			$story->bearcap_token = null;
			$story->path = $newPath;
			$story->save();
			Storage::deleteDirectory($dir);
		}
	}

	protected function fanoutExpiry()
	{
		$story = $this->story;
		$profile = $story->profile;

		if($story->local == false || $story->remote_url) {
			return;
		}

		$audience = FollowerService::softwareAudience($story->profile_id, 'pixelfed');

		if(empty($audience)) {
			// Return on profiles with no remote followers
			return;
		}

		$fractal = new Fractal\Manager();
		$fractal->setSerializer(new ArraySerializer());
		$resource = new Fractal\Resource\Item($story, new DeleteStory());
		$activity = $fractal->createData($resource)->toArray();

		$payload = json_encode($activity);

		$client = new Client([
			'timeout'  => config('federation.activitypub.delivery.timeout')
		]);

		$requests = function($audience) use ($client, $activity, $profile, $payload) {
			foreach($audience as $url) {
				$version = config('pixelfed.version');
				$appUrl = config('app.url');
				$headers = HttpSignature::sign($profile, $url, $activity, [
					'Content-Type'	=> 'application/ld+json; profile="https://www.w3.org/ns/activitystreams"',
					'User-Agent'	=> "(Pixelfed/{$version}; +{$appUrl})",
				]);
				yield function() use ($client, $url, $headers, $payload) {
					return $client->postAsync($url, [
						'curl' => [
							CURLOPT_HTTPHEADER => $headers,
							CURLOPT_POSTFIELDS => $payload,
							CURLOPT_HEADER => true
						]
					]);
				};
			}
		};

		$pool = new Pool($client, $requests($audience), [
			'concurrency' => config('federation.activitypub.delivery.concurrency'),
			'fulfilled' => function ($response, $index) {
			},
			'rejected' => function ($reason, $index) {
			}
		]);

		$promise = $pool->promise();

		$promise->wait();
	}

	protected function handleRemoteExpiry()
	{
		$story = $this->story;
		$story->active = false;
		$story->save();

		$path = $story->path;

		if(Storage::exists($path) == true) {
			Storage::delete($path);
		}

		$story->views()->delete();
		$story->delete();
	}
}