<?php namespace App\Http\Controllers; use Illuminate\Http\Request; use App\Models\Portfolio; use Cache; use DB; use App\Status; use App\User; use App\Services\AccountService; use App\Services\StatusService; class PortfolioController extends Controller { const RSS_FEED_KEY = 'pf:portfolio:rss-feed:'; const CACHED_FEED_KEY = 'pf:portfolio:cached-feed:'; const RECENT_FEED_KEY = 'pf:portfolio:recent-feed:'; public function index(Request $request) { return view('portfolio.index'); } public function show(Request $request, $username) { $user = User::whereUsername($username)->first(); if(!$user) { return view('portfolio.404'); } $portfolio = Portfolio::whereUserId($user->id)->firstOrFail(); $user = AccountService::get($user->profile_id); if($user['locked']) { return view('portfolio.404'); } if($portfolio->active != true) { if(!$request->user()) { return view('portfolio.404'); } if($request->user()->profile_id == $user['id']) { return redirect(config('portfolio.path') . '/settings'); } return view('portfolio.404'); } return view('portfolio.show', compact('user', 'portfolio')); } public function showPost(Request $request, $username, $id) { $authed = $request->user(); $post = StatusService::get($id); if(!$post) { return view('portfolio.404'); } $user = AccountService::get($post['account']['id']); $portfolio = Portfolio::whereProfileId($user['id'])->first(); if(!$portfolio || $user['locked'] || $portfolio->active != true) { return view('portfolio.404'); } if(!$post || $post['visibility'] != 'public' || !in_array($post['pf_type'], ['photo', 'photo:album']) || $user['id'] != $post['account']['id']) { return view('portfolio.404'); } return view('portfolio.show_post', compact('user', 'post', 'authed')); } public function myRedirect(Request $request) { abort_if(!$request->user(), 404); $user = $request->user(); if(Portfolio::whereProfileId($user->profile_id)->exists() === false) { $portfolio = new Portfolio; $portfolio->profile_id = $user->profile_id; $portfolio->user_id = $user->id; $portfolio->active = false; $portfolio->save(); } $domain = config('portfolio.domain'); $path = config('portfolio.path'); $url = 'https://' . $domain . $path; return redirect($url); } public function settings(Request $request) { if(!$request->user()) { return redirect(route('home')); } $portfolio = Portfolio::whereUserId($request->user()->id)->first(); if(!$portfolio) { $portfolio = new Portfolio; $portfolio->user_id = $request->user()->id; $portfolio->profile_id = $request->user()->profile_id; $portfolio->save(); } return view('portfolio.settings', compact('portfolio')); } public function store(Request $request) { abort_unless($request->user(), 404); $this->validate($request, [ 'profile_source' => 'required|in:recent,custom', 'layout' => 'required|in:grid,masonry', 'layout_container' => 'required|in:fixed,fluid', ]); $portfolio = Portfolio::whereUserId($request->user()->id)->first(); if(!$portfolio) { $portfolio = new Portfolio; $portfolio->user_id = $request->user()->id; $portfolio->profile_id = $request->user()->profile_id; $portfolio->save(); } $portfolio->active = $request->input('enabled') === 'on'; $portfolio->show_captions = $request->input('show_captions') === 'on'; $portfolio->show_license = $request->input('show_license') === 'on'; $portfolio->show_location = $request->input('show_location') === 'on'; $portfolio->show_timestamp = $request->input('show_timestamp') === 'on'; $portfolio->show_link = $request->input('show_link') === 'on'; $portfolio->profile_source = $request->input('profile_source'); $portfolio->show_avatar = $request->input('show_avatar') === 'on'; $portfolio->show_bio = $request->input('show_bio') === 'on'; $portfolio->profile_layout = $request->input('layout'); $portfolio->profile_container = $request->input('layout_container'); $portfolio->metadata = $metadata; $portfolio->save(); return redirect('/' . $request->user()->username); } public function getFeed(Request $request, $id) { $user = AccountService::get($id, true); if(!$user || !isset($user['id'])) { return response()->json([], 404); } $portfolio = Portfolio::whereProfileId($user['id'])->first(); if(!$portfolio || !$portfolio->active) { return response()->json([], 404); } if($portfolio->profile_source === 'custom' && $portfolio->metadata) { return $this->getCustomFeed($portfolio); } return $this->getRecentFeed($user['id']); } protected function getCustomFeed($portfolio) { if(!isset($portfolio->metadata['posts']) || !$portfolio->metadata['posts']) { return response()->json([], 400); } $feed = Cache::remember(self::CACHED_FEED_KEY . $portfolio->profile_id, 86400, function() use($portfolio) { return collect($portfolio->metadata['posts'])->map(function($p) { return StatusService::get($p); }) ->filter(function($p) { return $p && isset($p['account']); }); }); if($portfolio->metadata && isset($portfolio->metadata['feed_order']) && $portfolio->metadata['feed_order'] === 'recent') { return $feed->reverse()->values(); } else { return $feed->values(); } } protected function getRecentFeed($id) { $media = Cache::remember(self::RECENT_FEED_KEY . $id, 3600, function() use($id) { return DB::table('media') ->whereProfileId($id) ->whereNotNull('status_id') ->groupBy('status_id') ->orderByDesc('id') ->take(50) ->pluck('status_id'); }); return $media->map(function($sid) use($id) { return StatusService::get($sid); }) ->filter(function($post) { return $post && isset($post['media_attachments']) && !empty($post['media_attachments']) && $post['pf_type'] === 'photo' && $post['visibility'] === 'public'; }) ->take(24) ->values(); } public function getSettings(Request $request) { abort_if(!$request->user(), 403); $res = Portfolio::whereUserId($request->user()->id)->get(); if(!$res) { return []; } return $res->map(function($p) { $metadata = $p->metadata; $bgColor = $metadata && isset($metadata['background_color']) ? $metadata['background_color'] : '#000000'; $textColor = $metadata && isset($metadata['text_color']) ? $metadata['text_color'] : '#d4d4d8'; $rssEnabled = $metadata && isset($metadata['rss_enabled']) ? $metadata['rss_enabled'] : false; $rssButton = $metadata && isset($metadata['show_rss_button']) ? $metadata['show_rss_button'] : false; $colorScheme = $metadata && isset($metadata['color_scheme']) ? $metadata['color_scheme'] : 'dark'; $feedOrder = $metadata && isset($metadata['feed_order']) ? $metadata['feed_order'] : 'oldest'; return [ 'url' => $p->url(), 'pid' => (string) $p->profile_id, 'active' => (bool) $p->active, 'show_captions' => (bool) $p->show_captions, 'show_license' => (bool) $p->show_license, 'show_location' => (bool) $p->show_location, 'show_timestamp' => (bool) $p->show_timestamp, 'show_link' => (bool) $p->show_link, 'show_avatar' => (bool) $p->show_avatar, 'show_bio' => (bool) $p->show_bio, 'profile_layout' => $p->profile_layout, 'profile_source' => $p->profile_source, 'color_scheme' => $colorScheme, 'background_color' => $bgColor, 'text_color' => $textColor, 'show_profile_button' => true, 'rss_enabled' => $rssEnabled, 'show_rss_button' => $rssButton, 'feed_order' => $feedOrder, 'metadata' => $p->metadata ]; })->first(); } public function getAccountSettings(Request $request) { $this->validate($request, [ 'id' => 'required|integer' ]); $account = AccountService::get($request->input('id')); abort_if(!$account, 404); $p = Portfolio::whereProfileId($request->input('id'))->whereActive(1)->firstOrFail(); if(!$p) { return []; } $metadata = $p->metadata; $rssEnabled = $metadata && isset($metadata['rss_enabled']) ? $metadata['rss_enabled'] : false; $rssButton = $metadata && isset($metadata['show_rss_button']) ? $metadata['show_rss_button'] : false; $profileButton = $metadata && isset($metadata['show_profile_button']) ? $metadata['show_profile_button'] : false; $res = [ 'url' => $p->url(), 'show_captions' => (bool) $p->show_captions, 'show_license' => (bool) $p->show_license, 'show_location' => (bool) $p->show_location, 'show_timestamp' => (bool) $p->show_timestamp, 'show_link' => (bool) $p->show_link, 'show_avatar' => (bool) $p->show_avatar, 'show_bio' => (bool) $p->show_bio, 'profile_layout' => $p->profile_layout, 'profile_source' => $p->profile_source, 'show_profile_button' => $profileButton, 'rss_enabled' => $rssEnabled, 'show_rss_button' => $rssButton, ]; if($rssEnabled) { $res['rss_feed_url'] = $p->permalink('.rss'); } if($p->metadata) { if(isset($p->metadata['background_color'])) { $res['background_color'] = $p->metadata['background_color']; } if(isset($p->metadata['text_color'])) { $res['text_color'] = $p->metadata['text_color']; } } return $res; } public function storeSettings(Request $request) { abort_if(!$request->user(), 403); $this->validate($request, [ 'active' => 'sometimes|boolean', 'show_captions' => 'sometimes|boolean', 'show_license' => 'sometimes|boolean', 'show_location' => 'sometimes|boolean', 'show_timestamp' => 'sometimes|boolean', 'show_link' => 'sometimes|boolean', 'show_avatar' => 'sometimes|boolean', 'show_bio' => 'sometimes|boolean', 'profile_layout' => 'sometimes|in:grid,masonry,album', 'profile_source' => 'sometimes|in:recent,custom', 'color_scheme' => 'sometimes|in:light,dark,custom', 'show_profile_button' => 'sometimes|boolean', 'rss_enabled' => 'sometimes|boolean', 'show_rss_button' => 'sometimes|boolean', 'feed_order' => 'sometimes|in:oldest,recent', 'background_color' => [ 'sometimes', 'nullable', 'regex:/^#([a-f0-9]{6}|[a-f0-9]{3})$/i' ], 'text_color' => [ 'sometimes', 'nullable', 'regex:/^#([a-f0-9]{6}|[a-f0-9]{3})$/i' ], ]); $res = Portfolio::whereUserId($request->user()->id)->firstOrFail(); $pid = $request->user()->profile_id; $metadata = $res->metadata; $clearFeedCache = false; if($request->has('color_scheme')) { $metadata['color_scheme'] = $request->input('color_scheme'); } if($request->has('background_color')) { $metadata['background_color'] = $request->input('background_color'); $bgc = $request->background_color; if($bgc && $bgc !== '#000000') { $metadata['color_scheme'] = 'custom'; } } if($request->has('text_color')) { $metadata['text_color'] = $request->input('text_color'); $txc = $request->text_color; if($txc && $txc !== '#d4d4d8') { $metadata['color_scheme'] = 'custom'; } } if($request->has('show_profile_button')) { $metadata['show_profile_button'] = $request->input('show_profile_button'); } if($request->has('rss_enabled')) { $metadata['rss_enabled'] = $request->input('rss_enabled'); } if($request->has('show_rss_button')) { $metadata['show_rss_button'] = $metadata['rss_enabled'] ? $request->input('show_rss_button') : false; } if($request->has('feed_order')) { $metadata['feed_order'] = $request->input('feed_order'); } if(isset($metadata['background_color']) || isset($metadata['text_color'])) { $bgc = isset($metadata['background_color']) ? $metadata['background_color'] : null; $txc = isset($metadata['text_color']) ? $metadata['text_color'] : null; if((!$bgc || $bgc == '#000000') && (!$txc || $txc === '#d4d4d8') && $request->color_scheme != 'light') { $metadata['color_scheme'] = 'dark'; } } if($request->has('color_scheme') && $request->color_scheme === 'light') { $metadata['background_color'] = '#ffffff'; $metadata['text_color'] = '#000000'; $metadata['color_scheme'] = 'light'; } if($request->metadata !== $metadata) { $res->metadata = $metadata; $res->save(); } if($request->profile_layout != $res->profile_layout) { $clearFeedCache = true; } $res->update($request->only([ 'active', 'show_captions', 'show_license', 'show_location', 'show_timestamp', 'show_link', 'show_avatar', 'show_bio', 'profile_layout', 'profile_source' ])); Cache::forget(self::RECENT_FEED_KEY . $pid); if($clearFeedCache) { Cache::forget(self::RSS_FEED_KEY . $pid); } return 200; } public function storeCurated(Request $request) { abort_if(!$request->user(), 403); $this->validate($request, [ 'ids' => 'required|array|max:100' ]); $pid = $request->user()->profile_id; $ids = $request->input('ids'); Status::whereProfileId($pid) ->whereScope('public') ->whereIn('type', ['photo', 'photo:album']) ->findOrFail($ids); $p = Portfolio::whereProfileId($pid)->firstOrFail(); $metadata = $p->metadata; $metadata['posts'] = $ids; $p->metadata = $metadata; $p->save(); Cache::forget(self::RECENT_FEED_KEY . $pid); Cache::forget(self::RSS_FEED_KEY . $pid); Cache::forget(self::CACHED_FEED_KEY . $pid); return $request->ids; } public function getRssFeed(Request $request, $username) { $user = User::whereUsername($username)->first(); if(!$user) { return view('portfolio.404'); } $portfolio = Portfolio::whereUserId($user->id)->where('active', 1)->firstOrFail(); $metadata = $portfolio->metadata; abort_if(!$metadata || !isset($metadata['rss_enabled']), 404); abort_unless($metadata['rss_enabled'], 404); $account = AccountService::get($user->profile_id); $portfolioUrl = $portfolio->url(); $portfolioLayout = $portfolio->profile_layout; if(!isset($metadata['posts']) || !count($metadata['posts'])) { $feed = []; } else { $feed = Cache::remember( self::RSS_FEED_KEY . $user->profile_id, 43200, function() use($portfolio, $portfolioUrl, $portfolioLayout) { return collect($portfolio->metadata['posts'])->map(function($post) { return StatusService::get($post); }) ->filter() ->values() ->map(function($post, $idx) use($portfolioLayout, $portfolioUrl) { $ts = now()->parse($post['created_at']); $url = $portfolioLayout == 'album' ? $portfolioUrl . '?slide=' . ($idx + 1) : $portfolioUrl . '/' . $post['id']; return [ 'title' => 'Post by ' . $post['account']['username'] . ' on ' . $ts->format('D, d M Y'), 'description' => $post['content_text'], 'pubDate' => date('D, d M Y H:i:s ', strtotime($post['created_at'])) . 'GMT', 'url' => $url ]; }) ->reverse() ->take(10) ->toArray(); } ); } $now = date('D, d M Y H:i:s ') . 'GMT'; return response() ->view('portfolio.rss_feed', compact('account', 'now', 'feed', 'portfolioUrl'), 200) ->header('Content-Type', 'text/xml'); return response($feed)->withHeaders(['Content-Type' => 'text/xml']); } public function getApFeed(Request $request, $username) { $user = User::whereUsername($username)->first(); if(!$user) { return view('portfolio.404'); } $portfolio = Portfolio::whereUserId($user->id)->where('active', 1)->firstOrFail(); $metadata = $portfolio->metadata; $baseUrl = config('app.url'); $page = $request->input('page'); $res = [ '@context' => 'https://www.w3.org/ns/activitystreams', 'id' => $portfolio->permalink('.json'), 'type' => 'OrderedCollection', 'totalItems' => isset($metadata['posts']) ? count($metadata['posts']) : 0, ]; if($request->has('page')) { $start = $page == 1 ? 0 : ($page * 10 - 10); $res['id'] = $portfolio->permalink('.json?page=' . $page); $res['type'] = 'OrderedCollectionPage'; $res['next'] = $portfolio->permalink('.json?page=' . $page + 1); $res['partOf'] = $portfolio->permalink('.json'); $res['orderedItems'] = collect($metadata['posts'])->slice($start)->take(10)->map(function($p) { return StatusService::get($p); }) ->filter() ->map(function($p) { return $p['url']; }) ->values(); if(!$res['orderedItems'] || $res['orderedItems']->count() != 10) { unset($res['next']); } } else { $res['first'] = $portfolio->permalink('.json?page=1'); } return response()->json($res, 200, [], JSON_UNESCAPED_SLASHES|JSON_PRETTY_PRINT) ->header('Content-Type', 'application/activity+json'); } }